2016-06-14 13:55:28 +02:00
|
|
|
/* Copyright (c) 2003, 2016, Oracle and/or its affiliates.
|
|
|
|
Copyright (c) 2011, 2016, MariaDB
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
2006-12-23 20:17:15 +01:00
|
|
|
the Free Software Foundation; version 2 of the License.
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
2019-05-11 20:29:06 +02:00
|
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1335 USA */
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
/**
|
|
|
|
@file
|
|
|
|
|
|
|
|
@brief
|
|
|
|
This file defines all spatial functions
|
|
|
|
*/
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2005-05-26 12:09:14 +02:00
|
|
|
#ifdef USE_PRAGMA_IMPLEMENTATION
|
2003-05-30 12:22:34 +02:00
|
|
|
#pragma implementation // gcc: Class implementation
|
|
|
|
#endif
|
|
|
|
|
2017-06-18 05:42:16 +02:00
|
|
|
#include "mariadb.h"
|
2010-03-31 16:05:33 +02:00
|
|
|
#include "sql_priv.h"
|
|
|
|
/*
|
|
|
|
It is necessary to include set_var.h instead of item.h because there
|
|
|
|
are dependencies on include order for set_var.h and item.h. This
|
|
|
|
will be resolved later.
|
|
|
|
*/
|
|
|
|
#include "sql_class.h" // THD, set_var.h: THD
|
|
|
|
#include "set_var.h"
|
2004-01-23 16:54:22 +01:00
|
|
|
#ifdef HAVE_SPATIAL
|
2003-05-30 12:22:34 +02:00
|
|
|
#include <m_ctype.h>
|
2015-08-13 12:25:51 +02:00
|
|
|
#include "opt_range.h"
|
2019-07-09 17:47:57 +02:00
|
|
|
#include "item_geofunc.h"
|
2019-10-16 14:22:04 +02:00
|
|
|
#include "item_create.h"
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
|
2018-05-08 15:26:26 +02:00
|
|
|
bool Item_geometry_func::fix_length_and_dec()
|
2004-09-22 19:36:53 +02:00
|
|
|
{
|
|
|
|
collation.set(&my_charset_bin);
|
|
|
|
decimals=0;
|
MDEV-9217 Split Item::tmp_table_field_from_field_type() into virtual methods in Type_handler
- Adding Type_handler::make_table_field() and moving pieces of the code
from Item::tmp_table_field_from_field_type() to virtual implementations
for various type handlers.
- Adding a new Type_all_attributes, to access to Item's extended
attributes, such as decimal_precision() and geometry_type().
- Adding a new class Record_addr, to pass record related information
to Type_handler methods (ptr, null_ptr and null_bit) as a single structure.
Note, later it will possibly be extended for BIT-alike field purposes,
by adding new members (bit_ptr_arg, bit_ofs_arg).
- Moving the code from Field_new_decimal::create_from_item()
to Type_handler_newdecimal::make_table_field().
- Removing Field_new_decimal() and Field_geom() helper constructor
variants that were used for temporary field creation.
- Adding Item_field::type_handler(), Field::type_handler() and
Field_blob::type_handler() to return correct type handlers for
blob variants, according to Field_blob::packlength.
- Adding Type_handler_blob_common, as a common parent for
Type_handler_tiny_blob, Type_handler_blob, Type_handler_medium_blob
and Type_handler_long_blob.
- Implementing Type_handler_blob_common::Item_hybrid_func_fix_attributes().
It's needed for cases when TEXT variants of different character sets are mixed
in LEAST, GREATEST, CASE and its abreviations (IF, IFNULL, COALESCE), e.g.:
CREATE TABLE t1 (
a TINYTEXT CHARACTER SET latin1,
b TINYTEXT CHARACTER SET utf8
);
CREATE TABLE t2 AS SELECT COALESCE(a,b) FROM t1;
Type handler aggregation returns TINYTEXT as a common data type
for the two columns. But as conversion from latin1 to utf8
happens for "a", the maximum possible length of "a" grows from 255 to 255*3.
Type_handler_blob_common::Item_hybrid_func_fix_attributes() makes sure
to update the blob type handler according to max_length.
- Adding Type_handler::blob_type_handler(uint max_octet_length).
- Adding a few m_type_aggregator_for_result.add() pairs, because
now Item_xxx::type_handler() can return pointers to type_handler_tiny_blob,
type_handler_blob, type_handler_medium_blob, type_handler_long_blob.
Before the patch only type_handler_blob was possible result of type_handler().
- Making type_handler_tiny_blob, type_handler_blob, type_handler_medium_blob,
type_handler_long_blob public.
- Removing the condition in Item_sum_avg::create_tmp_field()
checking Item_sum_avg::result_type() against DECIMAL_RESULT.
Now both REAL_RESULT and DECIMAL_RESULT are symmetrically handled
by tmp_table_field_from_field_type().
- Removing Item_geometry_func::create_field_for_create_select(),
as the inherited version perfectly works.
- Fixing Item_func_as_wkb::field_type() to return MYSQL_TYPE_LONG_BLOB
rather than MYSQL_TYPE_BLOB. It's needed to make sure that
tmp_table_field_from_field_type() creates a LONGBLOB field for AsWKB().
- Fixing Item_func_as_wkt::fix_length_and_dec() to set max_length to
UINT32_MAX rather than MAX_BLOB_WIDTH, to make sure that
tmp_table_field_from_field_type() creates a LONGTEXT field for AsWKT().
- Removing Item_func_set_user_var::create_field_for_create_select(),
as the inherited version works fine.
- Adding Item_func_get_user_var::create_field_for_create_select() to
make sure that "CREATE TABLE t1 AS SELECT @string_user variable"
always creates a field of LONGTEXT/LONGBLOB type.
- Item_func_ifnull::create_field_for_create_select()
behavior has changed. Before the patch it passed set_blob_packflag=false,
which meant to create LONGBLOB for all blob variants.
Now it takes into account max_length, which gives better column
data types for:
CREATE TABLE t2 AS SELECT IFNULL(blob_column1, blob_column2) FROM t1;
- Fixing Item_func_nullif::fix_length_and_dec() to use
set_handler(args[2]->type_handler()) instead of
set_handler_by_field_type(args[2]->field_type()).
This is needed to distinguish between BLOB variants.
- Implementing Item_blob::type_handler(), to make sure to create
proper BLOB field variant, according to max_length, for queries like:
CREATE TABLE t1 AS
SELECT some_blob_field FROM INFORMATION_SCHEMA.SOME_TABLE;
- Fixing Item_field::real_type_handler() to make sure that
the code aggregating fields for UNION gets a proper BLOB
variant type handler from fields.
- Adding a special code into Item_type_holder::make_field_by_type(),
to make sure that after aggregating field types it also properly
takes into account max_length when mixing TEXT variants of different
character sets and chooses a proper TEXT variant:
CREATE TABLE t1 (
a TINYTEXT CHARACTER SET latin1,
b TINYTEXT CHARACTER SET utf8
);
CREATE TABLE t2 AS SELECT a FROM t1 UNION SELECT b FROM t1;
- Adding tests, for better coverage of IFNULL, NULLIF, UNION.
- The fact that tmp_table_field_from_field_type() now takes
into account BLOB variants (instead of always creating LONGBLOB),
tests results for WEIGHT_STRING() and NULLIF() and UNION
have become more precise.
2017-04-24 10:09:25 +02:00
|
|
|
max_length= (uint32) UINT_MAX32;
|
2013-09-25 14:30:13 +02:00
|
|
|
maybe_null= 1;
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2004-09-22 19:36:53 +02:00
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
String *Item_func_geometry_from_text::val_str(String *str)
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
2003-05-30 12:22:34 +02:00
|
|
|
String arg_val;
|
WL#2649 Number-to-string conversions
added:
include/ctype_numconv.inc
mysql-test/include/ctype_numconv.inc
mysql-test/r/ctype_binary.result
mysql-test/t/ctype_binary.test
Adding tests
modified:
mysql-test/r/bigint.result
mysql-test/r/case.result
mysql-test/r/create.result
mysql-test/r/ctype_cp1251.result
mysql-test/r/ctype_latin1.result
mysql-test/r/ctype_ucs.result
mysql-test/r/func_gconcat.result
mysql-test/r/func_str.result
mysql-test/r/metadata.result
mysql-test/r/ps_1general.result
mysql-test/r/ps_2myisam.result
mysql-test/r/ps_3innodb.result
mysql-test/r/ps_4heap.result
mysql-test/r/ps_5merge.result
mysql-test/r/show_check.result
mysql-test/r/type_datetime.result
mysql-test/r/type_ranges.result
mysql-test/r/union.result
mysql-test/suite/ndb/r/ps_7ndb.result
mysql-test/t/ctype_cp1251.test
mysql-test/t/ctype_latin1.test
mysql-test/t/ctype_ucs.test
mysql-test/t/func_str.test
Fixing tests
@ sql/field.cc
- Return str result using my_charset_numeric.
- Using real multi-byte aware str_to_XXX functions
to handle tricky charset values propely (e.g. UCS2)
@ sql/field.h
- Changing derivation of non-string field types to DERIVATION_NUMERIC.
- Changing binary() for numeric/datetime fields to always
return TRUE even if charset is not my_charset_bin. We need
this to keep ha_base_keytype() return HA_KEYTYPE_BINARY.
- Adding BINARY_FLAG into some fields, because it's not
being set automatically anymore with
"my_charset_bin to my_charset_numeric" change.
- Changing derivation for numeric/datetime datatypes to a weaker
value, to make "SELECT concat('string', field)" use character
set of the string literal for the result of the function.
@ sql/item.cc
- Implementing generic val_str_ascii().
- Using max_char_length() instead of direct read of max_length
to make "tricky" charsets like UCS2 work.
NOTE: in the future we'll possibly remove all direct reads of max_length
- Fixing Item_num::safe_charset_converter().
Previously it alligned binary string to
character string (for example by adding leading 0x00
when doing binary->UCS2 conversion). Now it just
converts from my_charset_numbner to "tocs".
- Using val_str_ascii() in Item::get_time() to make UCS2 arguments work.
- Other misc changes
@ sql/item.h
- Changing MY_COLL_CMP_CONV and MY_COLL_ALLOW_CONV to
bit operations instead of hard-coded bit masks.
- Addding new method DTCollation.set_numeric().
- Adding new methods to Item.
- Adding helper functions to make code look nicer:
agg_item_charsets_for_string_result()
agg_item_charsets_for_comparison()
- Changing charset for Item_num-derived items
from my_charset_bin to my_charset_numeric
(which is an alias for latin1).
@ sql/item_cmpfunc.cc
- Using new helper functions
- Other misc changes
@ sql/item_cmpfunc.h
- Fixing strcmp() to return max_length=2.
Previously it returned 1, which was wrong,
because it did not fit '-1'.
@ sql/item_func.cc
- Using new helper functions
- Other minor changes
@ sql/item_func.h
- Removing unused functions
- Adding helper functions
agg_arg_charsets_for_string_result()
agg_arg_charsets_for_comparison()
- Adding set_numeric() into constructors of numeric items.
- Using fix_length_and_charset() and fix_char_length()
instead of direct write to max_length.
@ sql/item_geofunc.cc
- Changing class for Item_func_geometry_type and
Item_func_as_wkt from Item_str_func to
Item_str_ascii_func, to make them return UCS2 result
properly (when character_set_connection=ucs2).
@ sql/item_geofunc.h
- Changing class for Item_func_geometry_type and
Item_func_as_wkt from Item_str_func to
Item_str_ascii_func, to make them return UCS2 result
properly (when @@character_set_connection=ucs2).
@ sql/item_strfunc.cc
- Implementing Item_str_func::val_str().
- Renaming val_str to val_str_ascii for some items,
to make them work with UCS2 properly.
- Using new helper functions
- All single-argument functions that expect string
result now call this method:
agg_arg_charsets_for_string_result(collation, args, 1);
This enables character set conversion to @@character_set_connection
in case of pure numeric input.
@ sql/item_strfunc.h
- Introducing Item_str_ascii_func - for functions
which return pure ASCII data, for performance purposes,
as well as for the cases when the old implementation
of val_str() was heavily 8-bit oriented and implementing
a UCS2-aware version is tricky.
@ sql/item_sum.cc
- Using new helper functions.
@ sql/item_timefunc.cc
- Using my_charset_numeric instead of my_charset_bin.
- Using fix_char_length(), fix_length_and_charset()
and fix_length_and_charset_datetime()
instead of direct write to max_length.
- Using tricky-charset aware function str_to_time_with_warn()
@ sql/item_timefunc.h
- Using new helper functions for charset and length initialization.
- Changing base class for Item_func_get_format() to make
it return UCS2 properly (when character_set_connection=ucs2).
@ sql/item_xmlfunc.cc
- Using new helper function
@ sql/my_decimal.cc
- Adding a new DECIMAL to CHAR converter
with real multibyte support (e.g. UCS2)
@ sql/mysql_priv.h
- Introducing a new derivation level for numeric/datetime data types.
- Adding macros for my_charset_numeric and MY_REPERTOIRE_NUMERIC.
- Adding prototypes for str_set_decimal()
- Adding prototypes for character-set aware str_to_xxx() functions.
@ sql/protocol.cc
- Changing charsetnr to "binary" client-side metadata for
numeric/datetime data types.
@ sql/time.cc
- Adding to_ascii() helper function, to convert a string
in any character set to ascii representation. In the
future can be extended to understand digits written
in various non-Latin word scripts.
- Adding real multy-byte character set aware versions for str_to_XXXX,
to make these these type of queries work correct:
INSERT INTO t1 SET datetime_column=ucs2_expression;
@ strings/ctype-ucs2.c
- endptr was not calculated correctly. INSERTing of UCS2
values into numeric columns returned warnings about
truncated wrong data.
2010-02-11 05:17:25 +01:00
|
|
|
String *wkt= args[0]->val_str_ascii(&arg_val);
|
2004-03-15 13:32:53 +01:00
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
Gis_read_stream trs(wkt->charset(), wkt->ptr(), wkt->length());
|
2004-03-04 07:50:37 +01:00
|
|
|
uint32 srid= 0;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
if ((arg_count == 2) && !args[1]->null_value)
|
2003-06-04 05:59:32 +02:00
|
|
|
srid= (uint32)args[1]->val_int();
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-09-07 13:44:25 +02:00
|
|
|
str->set_charset(&my_charset_bin);
|
2017-10-11 09:57:26 +02:00
|
|
|
str->length(0);
|
2003-05-30 12:22:34 +02:00
|
|
|
if (str->reserve(SRID_SIZE, 512))
|
|
|
|
return 0;
|
|
|
|
str->q_append(srid);
|
2007-03-02 12:09:44 +01:00
|
|
|
if ((null_value= !Geometry::create_from_wkt(&buffer, &trs, str, 0)))
|
|
|
|
return 0;
|
2003-05-30 12:22:34 +02:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_geometry_from_wkb::val_str(String *str)
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-30 12:22:34 +02:00
|
|
|
String arg_val;
|
2009-04-28 11:47:26 +02:00
|
|
|
String *wkb;
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
2004-03-04 07:50:37 +01:00
|
|
|
uint32 srid= 0;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2009-04-28 11:47:26 +02:00
|
|
|
if (args[0]->field_type() == MYSQL_TYPE_GEOMETRY)
|
|
|
|
{
|
2009-10-21 10:43:45 +02:00
|
|
|
String *str_ret= args[0]->val_str(str);
|
|
|
|
null_value= args[0]->null_value;
|
|
|
|
return str_ret;
|
2009-04-28 11:47:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
wkb= args[0]->val_str(&arg_val);
|
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
if ((arg_count == 2) && !args[1]->null_value)
|
2003-06-04 05:59:32 +02:00
|
|
|
srid= (uint32)args[1]->val_int();
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-09-07 13:44:25 +02:00
|
|
|
str->set_charset(&my_charset_bin);
|
2003-05-30 12:22:34 +02:00
|
|
|
if (str->reserve(SRID_SIZE, 512))
|
2009-10-21 10:43:45 +02:00
|
|
|
{
|
|
|
|
null_value= TRUE; /* purecov: inspected */
|
|
|
|
return 0; /* purecov: inspected */
|
|
|
|
}
|
2003-05-30 12:22:34 +02:00
|
|
|
str->length(0);
|
|
|
|
str->q_append(srid);
|
2004-03-12 09:04:00 +01:00
|
|
|
if ((null_value=
|
2009-04-28 11:47:26 +02:00
|
|
|
(args[0]->null_value ||
|
|
|
|
!Geometry::create_from_wkb(&buffer, wkb->ptr(), wkb->length(), str))))
|
2003-05-30 12:22:34 +02:00
|
|
|
return 0;
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-01-23 23:29:04 +01:00
|
|
|
void report_json_error_ex(String *js, json_engine_t *je,
|
|
|
|
const char *fname, int n_param,
|
|
|
|
Sql_condition::enum_warning_level lv);
|
|
|
|
|
|
|
|
String *Item_func_geometry_from_json::val_str(String *str)
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
|
|
|
Geometry_buffer buffer;
|
|
|
|
String *js= args[0]->val_str_ascii(&tmp_js);
|
|
|
|
uint32 srid= 0;
|
2017-08-06 14:27:37 +02:00
|
|
|
longlong options= 0;
|
2017-01-23 23:29:04 +01:00
|
|
|
json_engine_t je;
|
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
2017-08-06 14:27:37 +02:00
|
|
|
if (arg_count > 1 && !args[1]->null_value)
|
|
|
|
{
|
|
|
|
options= args[1]->val_int();
|
|
|
|
if (options > 4 || options < 1)
|
|
|
|
{
|
|
|
|
String *sv= args[1]->val_str(&tmp_js);
|
|
|
|
my_error(ER_WRONG_VALUE_FOR_TYPE, MYF(0),
|
2017-11-02 14:40:27 +01:00
|
|
|
"option", sv->c_ptr_safe(), "ST_GeometryFromJSON");
|
2017-08-06 14:27:37 +02:00
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((arg_count == 3) && !args[2]->null_value)
|
|
|
|
srid= (uint32)args[2]->val_int();
|
2017-01-23 23:29:04 +01:00
|
|
|
|
|
|
|
str->set_charset(&my_charset_bin);
|
|
|
|
if (str->reserve(SRID_SIZE, 512))
|
|
|
|
return 0;
|
|
|
|
str->length(0);
|
|
|
|
str->q_append(srid);
|
|
|
|
|
|
|
|
json_scan_start(&je, js->charset(), (const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->end());
|
|
|
|
|
2017-08-06 14:27:37 +02:00
|
|
|
if ((null_value= !Geometry::create_from_json(&buffer, &je, options==1, str)))
|
2017-01-23 23:29:04 +01:00
|
|
|
{
|
|
|
|
int code= 0;
|
|
|
|
|
|
|
|
switch (je.s.error)
|
|
|
|
{
|
|
|
|
case Geometry::GEOJ_INCORRECT_GEOJSON:
|
|
|
|
code= ER_GEOJSON_INCORRECT;
|
|
|
|
break;
|
|
|
|
case Geometry::GEOJ_TOO_FEW_POINTS:
|
|
|
|
code= ER_GEOJSON_TOO_FEW_POINTS;
|
|
|
|
break;
|
2017-11-14 10:36:50 +01:00
|
|
|
case Geometry::GEOJ_EMPTY_COORDINATES:
|
|
|
|
code= ER_GEOJSON_EMPTY_COORDINATES;
|
|
|
|
break;
|
2017-01-23 23:29:04 +01:00
|
|
|
case Geometry::GEOJ_POLYGON_NOT_CLOSED:
|
|
|
|
code= ER_GEOJSON_NOT_CLOSED;
|
|
|
|
break;
|
2017-08-06 14:27:37 +02:00
|
|
|
case Geometry::GEOJ_DIMENSION_NOT_SUPPORTED:
|
|
|
|
my_error(ER_GIS_INVALID_DATA, MYF(0), "ST_GeometryFromJSON");
|
|
|
|
break;
|
2017-01-23 23:29:04 +01:00
|
|
|
default:
|
|
|
|
report_json_error_ex(js, &je, func_name(), 0, Sql_condition::WARN_LEVEL_WARN);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (code)
|
|
|
|
{
|
|
|
|
THD *thd= current_thd;
|
|
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN, code,
|
|
|
|
ER_THD(thd, code));
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
WL#2649 Number-to-string conversions
added:
include/ctype_numconv.inc
mysql-test/include/ctype_numconv.inc
mysql-test/r/ctype_binary.result
mysql-test/t/ctype_binary.test
Adding tests
modified:
mysql-test/r/bigint.result
mysql-test/r/case.result
mysql-test/r/create.result
mysql-test/r/ctype_cp1251.result
mysql-test/r/ctype_latin1.result
mysql-test/r/ctype_ucs.result
mysql-test/r/func_gconcat.result
mysql-test/r/func_str.result
mysql-test/r/metadata.result
mysql-test/r/ps_1general.result
mysql-test/r/ps_2myisam.result
mysql-test/r/ps_3innodb.result
mysql-test/r/ps_4heap.result
mysql-test/r/ps_5merge.result
mysql-test/r/show_check.result
mysql-test/r/type_datetime.result
mysql-test/r/type_ranges.result
mysql-test/r/union.result
mysql-test/suite/ndb/r/ps_7ndb.result
mysql-test/t/ctype_cp1251.test
mysql-test/t/ctype_latin1.test
mysql-test/t/ctype_ucs.test
mysql-test/t/func_str.test
Fixing tests
@ sql/field.cc
- Return str result using my_charset_numeric.
- Using real multi-byte aware str_to_XXX functions
to handle tricky charset values propely (e.g. UCS2)
@ sql/field.h
- Changing derivation of non-string field types to DERIVATION_NUMERIC.
- Changing binary() for numeric/datetime fields to always
return TRUE even if charset is not my_charset_bin. We need
this to keep ha_base_keytype() return HA_KEYTYPE_BINARY.
- Adding BINARY_FLAG into some fields, because it's not
being set automatically anymore with
"my_charset_bin to my_charset_numeric" change.
- Changing derivation for numeric/datetime datatypes to a weaker
value, to make "SELECT concat('string', field)" use character
set of the string literal for the result of the function.
@ sql/item.cc
- Implementing generic val_str_ascii().
- Using max_char_length() instead of direct read of max_length
to make "tricky" charsets like UCS2 work.
NOTE: in the future we'll possibly remove all direct reads of max_length
- Fixing Item_num::safe_charset_converter().
Previously it alligned binary string to
character string (for example by adding leading 0x00
when doing binary->UCS2 conversion). Now it just
converts from my_charset_numbner to "tocs".
- Using val_str_ascii() in Item::get_time() to make UCS2 arguments work.
- Other misc changes
@ sql/item.h
- Changing MY_COLL_CMP_CONV and MY_COLL_ALLOW_CONV to
bit operations instead of hard-coded bit masks.
- Addding new method DTCollation.set_numeric().
- Adding new methods to Item.
- Adding helper functions to make code look nicer:
agg_item_charsets_for_string_result()
agg_item_charsets_for_comparison()
- Changing charset for Item_num-derived items
from my_charset_bin to my_charset_numeric
(which is an alias for latin1).
@ sql/item_cmpfunc.cc
- Using new helper functions
- Other misc changes
@ sql/item_cmpfunc.h
- Fixing strcmp() to return max_length=2.
Previously it returned 1, which was wrong,
because it did not fit '-1'.
@ sql/item_func.cc
- Using new helper functions
- Other minor changes
@ sql/item_func.h
- Removing unused functions
- Adding helper functions
agg_arg_charsets_for_string_result()
agg_arg_charsets_for_comparison()
- Adding set_numeric() into constructors of numeric items.
- Using fix_length_and_charset() and fix_char_length()
instead of direct write to max_length.
@ sql/item_geofunc.cc
- Changing class for Item_func_geometry_type and
Item_func_as_wkt from Item_str_func to
Item_str_ascii_func, to make them return UCS2 result
properly (when character_set_connection=ucs2).
@ sql/item_geofunc.h
- Changing class for Item_func_geometry_type and
Item_func_as_wkt from Item_str_func to
Item_str_ascii_func, to make them return UCS2 result
properly (when @@character_set_connection=ucs2).
@ sql/item_strfunc.cc
- Implementing Item_str_func::val_str().
- Renaming val_str to val_str_ascii for some items,
to make them work with UCS2 properly.
- Using new helper functions
- All single-argument functions that expect string
result now call this method:
agg_arg_charsets_for_string_result(collation, args, 1);
This enables character set conversion to @@character_set_connection
in case of pure numeric input.
@ sql/item_strfunc.h
- Introducing Item_str_ascii_func - for functions
which return pure ASCII data, for performance purposes,
as well as for the cases when the old implementation
of val_str() was heavily 8-bit oriented and implementing
a UCS2-aware version is tricky.
@ sql/item_sum.cc
- Using new helper functions.
@ sql/item_timefunc.cc
- Using my_charset_numeric instead of my_charset_bin.
- Using fix_char_length(), fix_length_and_charset()
and fix_length_and_charset_datetime()
instead of direct write to max_length.
- Using tricky-charset aware function str_to_time_with_warn()
@ sql/item_timefunc.h
- Using new helper functions for charset and length initialization.
- Changing base class for Item_func_get_format() to make
it return UCS2 properly (when character_set_connection=ucs2).
@ sql/item_xmlfunc.cc
- Using new helper function
@ sql/my_decimal.cc
- Adding a new DECIMAL to CHAR converter
with real multibyte support (e.g. UCS2)
@ sql/mysql_priv.h
- Introducing a new derivation level for numeric/datetime data types.
- Adding macros for my_charset_numeric and MY_REPERTOIRE_NUMERIC.
- Adding prototypes for str_set_decimal()
- Adding prototypes for character-set aware str_to_xxx() functions.
@ sql/protocol.cc
- Changing charsetnr to "binary" client-side metadata for
numeric/datetime data types.
@ sql/time.cc
- Adding to_ascii() helper function, to convert a string
in any character set to ascii representation. In the
future can be extended to understand digits written
in various non-Latin word scripts.
- Adding real multy-byte character set aware versions for str_to_XXXX,
to make these these type of queries work correct:
INSERT INTO t1 SET datetime_column=ucs2_expression;
@ strings/ctype-ucs2.c
- endptr was not calculated correctly. INSERTing of UCS2
values into numeric columns returned warnings about
truncated wrong data.
2010-02-11 05:17:25 +01:00
|
|
|
String *Item_func_as_wkt::val_str_ascii(String *str)
|
2003-05-30 12:22:34 +02:00
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-30 12:22:34 +02:00
|
|
|
String arg_val;
|
|
|
|
String *swkb= args[0]->val_str(&arg_val);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom= NULL;
|
2004-03-04 07:50:37 +01:00
|
|
|
const char *dummy;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-03-12 09:04:00 +01:00
|
|
|
if ((null_value=
|
|
|
|
(args[0]->null_value ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer, swkb->ptr(), swkb->length())))))
|
2003-05-30 12:22:34 +02:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
str->length(0);
|
2012-08-09 18:25:47 +02:00
|
|
|
str->set_charset(&my_charset_latin1);
|
2004-03-12 09:04:00 +01:00
|
|
|
if ((null_value= geom->as_wkt(str, &dummy)))
|
2003-05-30 12:22:34 +02:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2018-05-08 15:26:26 +02:00
|
|
|
bool Item_func_as_wkt::fix_length_and_dec()
|
2003-05-30 12:22:34 +02:00
|
|
|
{
|
WL#2649 Number-to-string conversions
added:
include/ctype_numconv.inc
mysql-test/include/ctype_numconv.inc
mysql-test/r/ctype_binary.result
mysql-test/t/ctype_binary.test
Adding tests
modified:
mysql-test/r/bigint.result
mysql-test/r/case.result
mysql-test/r/create.result
mysql-test/r/ctype_cp1251.result
mysql-test/r/ctype_latin1.result
mysql-test/r/ctype_ucs.result
mysql-test/r/func_gconcat.result
mysql-test/r/func_str.result
mysql-test/r/metadata.result
mysql-test/r/ps_1general.result
mysql-test/r/ps_2myisam.result
mysql-test/r/ps_3innodb.result
mysql-test/r/ps_4heap.result
mysql-test/r/ps_5merge.result
mysql-test/r/show_check.result
mysql-test/r/type_datetime.result
mysql-test/r/type_ranges.result
mysql-test/r/union.result
mysql-test/suite/ndb/r/ps_7ndb.result
mysql-test/t/ctype_cp1251.test
mysql-test/t/ctype_latin1.test
mysql-test/t/ctype_ucs.test
mysql-test/t/func_str.test
Fixing tests
@ sql/field.cc
- Return str result using my_charset_numeric.
- Using real multi-byte aware str_to_XXX functions
to handle tricky charset values propely (e.g. UCS2)
@ sql/field.h
- Changing derivation of non-string field types to DERIVATION_NUMERIC.
- Changing binary() for numeric/datetime fields to always
return TRUE even if charset is not my_charset_bin. We need
this to keep ha_base_keytype() return HA_KEYTYPE_BINARY.
- Adding BINARY_FLAG into some fields, because it's not
being set automatically anymore with
"my_charset_bin to my_charset_numeric" change.
- Changing derivation for numeric/datetime datatypes to a weaker
value, to make "SELECT concat('string', field)" use character
set of the string literal for the result of the function.
@ sql/item.cc
- Implementing generic val_str_ascii().
- Using max_char_length() instead of direct read of max_length
to make "tricky" charsets like UCS2 work.
NOTE: in the future we'll possibly remove all direct reads of max_length
- Fixing Item_num::safe_charset_converter().
Previously it alligned binary string to
character string (for example by adding leading 0x00
when doing binary->UCS2 conversion). Now it just
converts from my_charset_numbner to "tocs".
- Using val_str_ascii() in Item::get_time() to make UCS2 arguments work.
- Other misc changes
@ sql/item.h
- Changing MY_COLL_CMP_CONV and MY_COLL_ALLOW_CONV to
bit operations instead of hard-coded bit masks.
- Addding new method DTCollation.set_numeric().
- Adding new methods to Item.
- Adding helper functions to make code look nicer:
agg_item_charsets_for_string_result()
agg_item_charsets_for_comparison()
- Changing charset for Item_num-derived items
from my_charset_bin to my_charset_numeric
(which is an alias for latin1).
@ sql/item_cmpfunc.cc
- Using new helper functions
- Other misc changes
@ sql/item_cmpfunc.h
- Fixing strcmp() to return max_length=2.
Previously it returned 1, which was wrong,
because it did not fit '-1'.
@ sql/item_func.cc
- Using new helper functions
- Other minor changes
@ sql/item_func.h
- Removing unused functions
- Adding helper functions
agg_arg_charsets_for_string_result()
agg_arg_charsets_for_comparison()
- Adding set_numeric() into constructors of numeric items.
- Using fix_length_and_charset() and fix_char_length()
instead of direct write to max_length.
@ sql/item_geofunc.cc
- Changing class for Item_func_geometry_type and
Item_func_as_wkt from Item_str_func to
Item_str_ascii_func, to make them return UCS2 result
properly (when character_set_connection=ucs2).
@ sql/item_geofunc.h
- Changing class for Item_func_geometry_type and
Item_func_as_wkt from Item_str_func to
Item_str_ascii_func, to make them return UCS2 result
properly (when @@character_set_connection=ucs2).
@ sql/item_strfunc.cc
- Implementing Item_str_func::val_str().
- Renaming val_str to val_str_ascii for some items,
to make them work with UCS2 properly.
- Using new helper functions
- All single-argument functions that expect string
result now call this method:
agg_arg_charsets_for_string_result(collation, args, 1);
This enables character set conversion to @@character_set_connection
in case of pure numeric input.
@ sql/item_strfunc.h
- Introducing Item_str_ascii_func - for functions
which return pure ASCII data, for performance purposes,
as well as for the cases when the old implementation
of val_str() was heavily 8-bit oriented and implementing
a UCS2-aware version is tricky.
@ sql/item_sum.cc
- Using new helper functions.
@ sql/item_timefunc.cc
- Using my_charset_numeric instead of my_charset_bin.
- Using fix_char_length(), fix_length_and_charset()
and fix_length_and_charset_datetime()
instead of direct write to max_length.
- Using tricky-charset aware function str_to_time_with_warn()
@ sql/item_timefunc.h
- Using new helper functions for charset and length initialization.
- Changing base class for Item_func_get_format() to make
it return UCS2 properly (when character_set_connection=ucs2).
@ sql/item_xmlfunc.cc
- Using new helper function
@ sql/my_decimal.cc
- Adding a new DECIMAL to CHAR converter
with real multibyte support (e.g. UCS2)
@ sql/mysql_priv.h
- Introducing a new derivation level for numeric/datetime data types.
- Adding macros for my_charset_numeric and MY_REPERTOIRE_NUMERIC.
- Adding prototypes for str_set_decimal()
- Adding prototypes for character-set aware str_to_xxx() functions.
@ sql/protocol.cc
- Changing charsetnr to "binary" client-side metadata for
numeric/datetime data types.
@ sql/time.cc
- Adding to_ascii() helper function, to convert a string
in any character set to ascii representation. In the
future can be extended to understand digits written
in various non-Latin word scripts.
- Adding real multy-byte character set aware versions for str_to_XXXX,
to make these these type of queries work correct:
INSERT INTO t1 SET datetime_column=ucs2_expression;
@ strings/ctype-ucs2.c
- endptr was not calculated correctly. INSERTing of UCS2
values into numeric columns returned warnings about
truncated wrong data.
2010-02-11 05:17:25 +01:00
|
|
|
collation.set(default_charset(), DERIVATION_COERCIBLE, MY_REPERTOIRE_ASCII);
|
MDEV-9217 Split Item::tmp_table_field_from_field_type() into virtual methods in Type_handler
- Adding Type_handler::make_table_field() and moving pieces of the code
from Item::tmp_table_field_from_field_type() to virtual implementations
for various type handlers.
- Adding a new Type_all_attributes, to access to Item's extended
attributes, such as decimal_precision() and geometry_type().
- Adding a new class Record_addr, to pass record related information
to Type_handler methods (ptr, null_ptr and null_bit) as a single structure.
Note, later it will possibly be extended for BIT-alike field purposes,
by adding new members (bit_ptr_arg, bit_ofs_arg).
- Moving the code from Field_new_decimal::create_from_item()
to Type_handler_newdecimal::make_table_field().
- Removing Field_new_decimal() and Field_geom() helper constructor
variants that were used for temporary field creation.
- Adding Item_field::type_handler(), Field::type_handler() and
Field_blob::type_handler() to return correct type handlers for
blob variants, according to Field_blob::packlength.
- Adding Type_handler_blob_common, as a common parent for
Type_handler_tiny_blob, Type_handler_blob, Type_handler_medium_blob
and Type_handler_long_blob.
- Implementing Type_handler_blob_common::Item_hybrid_func_fix_attributes().
It's needed for cases when TEXT variants of different character sets are mixed
in LEAST, GREATEST, CASE and its abreviations (IF, IFNULL, COALESCE), e.g.:
CREATE TABLE t1 (
a TINYTEXT CHARACTER SET latin1,
b TINYTEXT CHARACTER SET utf8
);
CREATE TABLE t2 AS SELECT COALESCE(a,b) FROM t1;
Type handler aggregation returns TINYTEXT as a common data type
for the two columns. But as conversion from latin1 to utf8
happens for "a", the maximum possible length of "a" grows from 255 to 255*3.
Type_handler_blob_common::Item_hybrid_func_fix_attributes() makes sure
to update the blob type handler according to max_length.
- Adding Type_handler::blob_type_handler(uint max_octet_length).
- Adding a few m_type_aggregator_for_result.add() pairs, because
now Item_xxx::type_handler() can return pointers to type_handler_tiny_blob,
type_handler_blob, type_handler_medium_blob, type_handler_long_blob.
Before the patch only type_handler_blob was possible result of type_handler().
- Making type_handler_tiny_blob, type_handler_blob, type_handler_medium_blob,
type_handler_long_blob public.
- Removing the condition in Item_sum_avg::create_tmp_field()
checking Item_sum_avg::result_type() against DECIMAL_RESULT.
Now both REAL_RESULT and DECIMAL_RESULT are symmetrically handled
by tmp_table_field_from_field_type().
- Removing Item_geometry_func::create_field_for_create_select(),
as the inherited version perfectly works.
- Fixing Item_func_as_wkb::field_type() to return MYSQL_TYPE_LONG_BLOB
rather than MYSQL_TYPE_BLOB. It's needed to make sure that
tmp_table_field_from_field_type() creates a LONGBLOB field for AsWKB().
- Fixing Item_func_as_wkt::fix_length_and_dec() to set max_length to
UINT32_MAX rather than MAX_BLOB_WIDTH, to make sure that
tmp_table_field_from_field_type() creates a LONGTEXT field for AsWKT().
- Removing Item_func_set_user_var::create_field_for_create_select(),
as the inherited version works fine.
- Adding Item_func_get_user_var::create_field_for_create_select() to
make sure that "CREATE TABLE t1 AS SELECT @string_user variable"
always creates a field of LONGTEXT/LONGBLOB type.
- Item_func_ifnull::create_field_for_create_select()
behavior has changed. Before the patch it passed set_blob_packflag=false,
which meant to create LONGBLOB for all blob variants.
Now it takes into account max_length, which gives better column
data types for:
CREATE TABLE t2 AS SELECT IFNULL(blob_column1, blob_column2) FROM t1;
- Fixing Item_func_nullif::fix_length_and_dec() to use
set_handler(args[2]->type_handler()) instead of
set_handler_by_field_type(args[2]->field_type()).
This is needed to distinguish between BLOB variants.
- Implementing Item_blob::type_handler(), to make sure to create
proper BLOB field variant, according to max_length, for queries like:
CREATE TABLE t1 AS
SELECT some_blob_field FROM INFORMATION_SCHEMA.SOME_TABLE;
- Fixing Item_field::real_type_handler() to make sure that
the code aggregating fields for UNION gets a proper BLOB
variant type handler from fields.
- Adding a special code into Item_type_holder::make_field_by_type(),
to make sure that after aggregating field types it also properly
takes into account max_length when mixing TEXT variants of different
character sets and chooses a proper TEXT variant:
CREATE TABLE t1 (
a TINYTEXT CHARACTER SET latin1,
b TINYTEXT CHARACTER SET utf8
);
CREATE TABLE t2 AS SELECT a FROM t1 UNION SELECT b FROM t1;
- Adding tests, for better coverage of IFNULL, NULLIF, UNION.
- The fact that tmp_table_field_from_field_type() now takes
into account BLOB variants (instead of always creating LONGBLOB),
tests results for WEIGHT_STRING() and NULLIF() and UNION
have become more precise.
2017-04-24 10:09:25 +02:00
|
|
|
max_length= (uint32) UINT_MAX32;
|
2013-09-25 14:30:13 +02:00
|
|
|
maybe_null= 1;
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
String *Item_func_as_wkb::val_str(String *str)
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-30 12:22:34 +02:00
|
|
|
String arg_val;
|
|
|
|
String *swkb= args[0]->val_str(&arg_val);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-03-12 09:04:00 +01:00
|
|
|
if ((null_value=
|
|
|
|
(args[0]->null_value ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(Geometry::construct(&buffer, swkb->ptr(), swkb->length())))))
|
2003-05-30 12:22:34 +02:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
str->copy(swkb->ptr() + SRID_SIZE, swkb->length() - SRID_SIZE,
|
|
|
|
&my_charset_bin);
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2018-05-08 15:26:26 +02:00
|
|
|
bool Item_func_as_geojson::fix_length_and_dec()
|
2017-01-23 23:29:04 +01:00
|
|
|
{
|
|
|
|
collation.set(default_charset(), DERIVATION_COERCIBLE, MY_REPERTOIRE_ASCII);
|
|
|
|
max_length=MAX_BLOB_WIDTH;
|
|
|
|
maybe_null= 1;
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2017-01-23 23:29:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_as_geojson::val_str_ascii(String *str)
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
|
|
|
String arg_val;
|
|
|
|
String *swkb= args[0]->val_str(&arg_val);
|
2017-08-07 10:49:04 +02:00
|
|
|
uint max_dec= FLOATING_POINT_DECIMALS;
|
|
|
|
longlong options= 0;
|
2017-01-23 23:29:04 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom= NULL;
|
|
|
|
const char *dummy;
|
|
|
|
|
|
|
|
if ((null_value=
|
|
|
|
(args[0]->null_value ||
|
|
|
|
!(geom= Geometry::construct(&buffer, swkb->ptr(), swkb->length())))))
|
|
|
|
return 0;
|
|
|
|
|
2017-08-07 10:49:04 +02:00
|
|
|
if (arg_count > 1)
|
|
|
|
{
|
|
|
|
max_dec= (uint) args[1]->val_int();
|
|
|
|
if (args[1]->null_value)
|
|
|
|
max_dec= FLOATING_POINT_DECIMALS;
|
|
|
|
if (arg_count > 2)
|
|
|
|
{
|
|
|
|
options= args[2]->val_int();
|
|
|
|
if (args[2]->null_value)
|
|
|
|
options= 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-23 23:29:04 +01:00
|
|
|
str->length(0);
|
|
|
|
str->set_charset(&my_charset_latin1);
|
2017-08-07 10:49:04 +02:00
|
|
|
|
|
|
|
if (str->reserve(1, 512))
|
2017-01-23 23:29:04 +01:00
|
|
|
return 0;
|
|
|
|
|
2017-08-07 10:49:04 +02:00
|
|
|
str->qs_append('{');
|
|
|
|
|
|
|
|
if (options & 1)
|
|
|
|
{
|
|
|
|
if (geom->bbox_as_json(str) || str->append(", ", 2))
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((geom->as_json(str, max_dec, &dummy) || str->append("}", 1)))
|
|
|
|
goto error;
|
|
|
|
|
2017-01-23 23:29:04 +01:00
|
|
|
return str;
|
2017-08-07 10:49:04 +02:00
|
|
|
|
|
|
|
error:
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
2017-01-23 23:29:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
WL#2649 Number-to-string conversions
added:
include/ctype_numconv.inc
mysql-test/include/ctype_numconv.inc
mysql-test/r/ctype_binary.result
mysql-test/t/ctype_binary.test
Adding tests
modified:
mysql-test/r/bigint.result
mysql-test/r/case.result
mysql-test/r/create.result
mysql-test/r/ctype_cp1251.result
mysql-test/r/ctype_latin1.result
mysql-test/r/ctype_ucs.result
mysql-test/r/func_gconcat.result
mysql-test/r/func_str.result
mysql-test/r/metadata.result
mysql-test/r/ps_1general.result
mysql-test/r/ps_2myisam.result
mysql-test/r/ps_3innodb.result
mysql-test/r/ps_4heap.result
mysql-test/r/ps_5merge.result
mysql-test/r/show_check.result
mysql-test/r/type_datetime.result
mysql-test/r/type_ranges.result
mysql-test/r/union.result
mysql-test/suite/ndb/r/ps_7ndb.result
mysql-test/t/ctype_cp1251.test
mysql-test/t/ctype_latin1.test
mysql-test/t/ctype_ucs.test
mysql-test/t/func_str.test
Fixing tests
@ sql/field.cc
- Return str result using my_charset_numeric.
- Using real multi-byte aware str_to_XXX functions
to handle tricky charset values propely (e.g. UCS2)
@ sql/field.h
- Changing derivation of non-string field types to DERIVATION_NUMERIC.
- Changing binary() for numeric/datetime fields to always
return TRUE even if charset is not my_charset_bin. We need
this to keep ha_base_keytype() return HA_KEYTYPE_BINARY.
- Adding BINARY_FLAG into some fields, because it's not
being set automatically anymore with
"my_charset_bin to my_charset_numeric" change.
- Changing derivation for numeric/datetime datatypes to a weaker
value, to make "SELECT concat('string', field)" use character
set of the string literal for the result of the function.
@ sql/item.cc
- Implementing generic val_str_ascii().
- Using max_char_length() instead of direct read of max_length
to make "tricky" charsets like UCS2 work.
NOTE: in the future we'll possibly remove all direct reads of max_length
- Fixing Item_num::safe_charset_converter().
Previously it alligned binary string to
character string (for example by adding leading 0x00
when doing binary->UCS2 conversion). Now it just
converts from my_charset_numbner to "tocs".
- Using val_str_ascii() in Item::get_time() to make UCS2 arguments work.
- Other misc changes
@ sql/item.h
- Changing MY_COLL_CMP_CONV and MY_COLL_ALLOW_CONV to
bit operations instead of hard-coded bit masks.
- Addding new method DTCollation.set_numeric().
- Adding new methods to Item.
- Adding helper functions to make code look nicer:
agg_item_charsets_for_string_result()
agg_item_charsets_for_comparison()
- Changing charset for Item_num-derived items
from my_charset_bin to my_charset_numeric
(which is an alias for latin1).
@ sql/item_cmpfunc.cc
- Using new helper functions
- Other misc changes
@ sql/item_cmpfunc.h
- Fixing strcmp() to return max_length=2.
Previously it returned 1, which was wrong,
because it did not fit '-1'.
@ sql/item_func.cc
- Using new helper functions
- Other minor changes
@ sql/item_func.h
- Removing unused functions
- Adding helper functions
agg_arg_charsets_for_string_result()
agg_arg_charsets_for_comparison()
- Adding set_numeric() into constructors of numeric items.
- Using fix_length_and_charset() and fix_char_length()
instead of direct write to max_length.
@ sql/item_geofunc.cc
- Changing class for Item_func_geometry_type and
Item_func_as_wkt from Item_str_func to
Item_str_ascii_func, to make them return UCS2 result
properly (when character_set_connection=ucs2).
@ sql/item_geofunc.h
- Changing class for Item_func_geometry_type and
Item_func_as_wkt from Item_str_func to
Item_str_ascii_func, to make them return UCS2 result
properly (when @@character_set_connection=ucs2).
@ sql/item_strfunc.cc
- Implementing Item_str_func::val_str().
- Renaming val_str to val_str_ascii for some items,
to make them work with UCS2 properly.
- Using new helper functions
- All single-argument functions that expect string
result now call this method:
agg_arg_charsets_for_string_result(collation, args, 1);
This enables character set conversion to @@character_set_connection
in case of pure numeric input.
@ sql/item_strfunc.h
- Introducing Item_str_ascii_func - for functions
which return pure ASCII data, for performance purposes,
as well as for the cases when the old implementation
of val_str() was heavily 8-bit oriented and implementing
a UCS2-aware version is tricky.
@ sql/item_sum.cc
- Using new helper functions.
@ sql/item_timefunc.cc
- Using my_charset_numeric instead of my_charset_bin.
- Using fix_char_length(), fix_length_and_charset()
and fix_length_and_charset_datetime()
instead of direct write to max_length.
- Using tricky-charset aware function str_to_time_with_warn()
@ sql/item_timefunc.h
- Using new helper functions for charset and length initialization.
- Changing base class for Item_func_get_format() to make
it return UCS2 properly (when character_set_connection=ucs2).
@ sql/item_xmlfunc.cc
- Using new helper function
@ sql/my_decimal.cc
- Adding a new DECIMAL to CHAR converter
with real multibyte support (e.g. UCS2)
@ sql/mysql_priv.h
- Introducing a new derivation level for numeric/datetime data types.
- Adding macros for my_charset_numeric and MY_REPERTOIRE_NUMERIC.
- Adding prototypes for str_set_decimal()
- Adding prototypes for character-set aware str_to_xxx() functions.
@ sql/protocol.cc
- Changing charsetnr to "binary" client-side metadata for
numeric/datetime data types.
@ sql/time.cc
- Adding to_ascii() helper function, to convert a string
in any character set to ascii representation. In the
future can be extended to understand digits written
in various non-Latin word scripts.
- Adding real multy-byte character set aware versions for str_to_XXXX,
to make these these type of queries work correct:
INSERT INTO t1 SET datetime_column=ucs2_expression;
@ strings/ctype-ucs2.c
- endptr was not calculated correctly. INSERTing of UCS2
values into numeric columns returned warnings about
truncated wrong data.
2010-02-11 05:17:25 +01:00
|
|
|
String *Item_func_geometry_type::val_str_ascii(String *str)
|
2003-05-30 12:22:34 +02:00
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-30 12:22:34 +02:00
|
|
|
String *swkb= args[0]->val_str(str);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom= NULL;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-03-12 09:04:00 +01:00
|
|
|
if ((null_value=
|
|
|
|
(args[0]->null_value ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer, swkb->ptr(), swkb->length())))))
|
2003-05-30 12:22:34 +02:00
|
|
|
return 0;
|
2004-03-04 07:50:37 +01:00
|
|
|
/* String will not move */
|
2004-10-22 17:32:02 +02:00
|
|
|
str->copy(geom->get_class_info()->m_name.str,
|
|
|
|
geom->get_class_info()->m_name.length,
|
2012-08-09 18:25:47 +02:00
|
|
|
&my_charset_latin1);
|
2003-05-30 12:22:34 +02:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_envelope::val_str(String *str)
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-30 12:22:34 +02:00
|
|
|
String arg_val;
|
|
|
|
String *swkb= args[0]->val_str(&arg_val);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom= NULL;
|
2004-03-04 07:50:37 +01:00
|
|
|
uint32 srid;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-03-12 09:04:00 +01:00
|
|
|
if ((null_value=
|
|
|
|
args[0]->null_value ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer, swkb->ptr(), swkb->length()))))
|
2003-05-30 12:22:34 +02:00
|
|
|
return 0;
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
srid= uint4korr(swkb->ptr());
|
2004-09-07 13:44:25 +02:00
|
|
|
str->set_charset(&my_charset_bin);
|
2003-05-30 12:22:34 +02:00
|
|
|
str->length(0);
|
|
|
|
if (str->reserve(SRID_SIZE, 512))
|
|
|
|
return 0;
|
|
|
|
str->q_append(srid);
|
2004-03-12 09:04:00 +01:00
|
|
|
return (null_value= geom->envelope(str)) ? 0 : str;
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-11-27 21:29:37 +01:00
|
|
|
int Item_func_boundary::Transporter::single_point(double x, double y)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_boundary::Transporter::start_line()
|
|
|
|
{
|
|
|
|
n_points= 0;
|
|
|
|
current_type= Gcalc_function::shape_line;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_boundary::Transporter::complete_line()
|
|
|
|
{
|
|
|
|
current_type= (Gcalc_function::shape_type) 0;
|
|
|
|
if (n_points > 1)
|
|
|
|
return m_receiver->single_point(last_x, last_y);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_boundary::Transporter::start_poly()
|
|
|
|
{
|
|
|
|
current_type= Gcalc_function::shape_polygon;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_boundary::Transporter::complete_poly()
|
|
|
|
{
|
|
|
|
current_type= (Gcalc_function::shape_type) 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_boundary::Transporter::start_ring()
|
|
|
|
{
|
|
|
|
n_points= 0;
|
|
|
|
return m_receiver->start_shape(Gcalc_function::shape_line);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_boundary::Transporter::complete_ring()
|
|
|
|
{
|
|
|
|
if (n_points > 1)
|
|
|
|
{
|
|
|
|
m_receiver->add_point(last_x, last_y);
|
|
|
|
}
|
|
|
|
m_receiver->complete_shape();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_boundary::Transporter::add_point(double x, double y)
|
|
|
|
{
|
|
|
|
++n_points;
|
|
|
|
if (current_type== Gcalc_function::shape_polygon)
|
|
|
|
{
|
|
|
|
/* Polygon's ring case */
|
|
|
|
if (n_points == 1)
|
|
|
|
{
|
|
|
|
last_x= x;
|
|
|
|
last_y= y;
|
|
|
|
}
|
|
|
|
return m_receiver->add_point(x, y);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (current_type== Gcalc_function::shape_line)
|
|
|
|
{
|
|
|
|
/* Line's case */
|
|
|
|
last_x= x;
|
|
|
|
last_y= y;
|
|
|
|
if (n_points == 1)
|
|
|
|
return m_receiver->single_point(x, y);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_boundary::Transporter::start_collection(int n_objects)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_boundary::val_str(String *str_value)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_func_boundary::val_str");
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
|
|
|
String arg_val;
|
|
|
|
String *swkb= args[0]->val_str(&arg_val);
|
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *g;
|
|
|
|
uint32 srid= 0;
|
|
|
|
Transporter trn(&res_receiver);
|
|
|
|
|
|
|
|
if ((null_value=
|
|
|
|
args[0]->null_value ||
|
|
|
|
!(g= Geometry::construct(&buffer, swkb->ptr(), swkb->length()))))
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
|
|
|
if (g->store_shapes(&trn))
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
str_value->set_charset(&my_charset_bin);
|
|
|
|
if (str_value->reserve(SRID_SIZE, 512))
|
|
|
|
goto mem_error;
|
|
|
|
str_value->length(0);
|
|
|
|
str_value->q_append(srid);
|
|
|
|
|
|
|
|
if (!Geometry::create_from_opresult(&buffer, str_value, res_receiver))
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
res_receiver.reset();
|
|
|
|
DBUG_RETURN(str_value);
|
|
|
|
|
|
|
|
mem_error:
|
|
|
|
null_value= 1;
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
String *Item_func_centroid::val_str(String *str)
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-30 12:22:34 +02:00
|
|
|
String arg_val;
|
|
|
|
String *swkb= args[0]->val_str(&arg_val);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom= NULL;
|
2004-03-04 07:50:37 +01:00
|
|
|
uint32 srid;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer, swkb->ptr(), swkb->length()))))
|
2003-05-30 12:22:34 +02:00
|
|
|
return 0;
|
|
|
|
|
2004-09-07 13:44:25 +02:00
|
|
|
str->set_charset(&my_charset_bin);
|
2003-05-30 12:22:34 +02:00
|
|
|
if (str->reserve(SRID_SIZE, 512))
|
|
|
|
return 0;
|
|
|
|
str->length(0);
|
2004-03-04 07:50:37 +01:00
|
|
|
srid= uint4korr(swkb->ptr());
|
2003-05-30 12:22:34 +02:00
|
|
|
str->q_append(srid);
|
|
|
|
|
2014-02-19 11:05:15 +01:00
|
|
|
return (null_value= MY_TEST(geom->centroid(str))) ? 0 : str;
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-11-27 21:29:37 +01:00
|
|
|
int Item_func_convexhull::add_node_to_line(ch_node **p_cur, int dir,
|
|
|
|
const Gcalc_heap::Info *pi)
|
|
|
|
{
|
|
|
|
ch_node *new_node;
|
|
|
|
ch_node *cur= *p_cur;
|
|
|
|
|
|
|
|
while (cur->prev)
|
|
|
|
{
|
|
|
|
int v_sign= Gcalc_scan_iterator::point::cmp_dx_dy(
|
|
|
|
cur->prev->pi, cur->pi, cur->pi, pi);
|
|
|
|
if (v_sign*dir <0)
|
|
|
|
break;
|
|
|
|
new_node= cur;
|
|
|
|
cur= cur->prev;
|
|
|
|
res_heap.free_item(new_node);
|
|
|
|
}
|
|
|
|
if (!(new_node= new_ch_node()))
|
|
|
|
return 1;
|
|
|
|
cur->next= new_node;
|
|
|
|
new_node->prev= cur;
|
|
|
|
new_node->pi= pi;
|
|
|
|
*p_cur= new_node;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef HEAVY_CONVEX_HULL
|
|
|
|
String *Item_func_convexhull::val_str(String *str_value)
|
|
|
|
{
|
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom= NULL;
|
|
|
|
MBR mbr;
|
|
|
|
const char *c_end;
|
|
|
|
Gcalc_operation_transporter trn(&func, &collector);
|
|
|
|
uint32 srid= 0;
|
|
|
|
ch_node *left_first, *left_cur, *right_first, *right_cur;
|
|
|
|
Gcalc_heap::Info *cur_pi;
|
|
|
|
|
|
|
|
DBUG_ENTER("Item_func_convexhull::val_str");
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
|
|
|
String *swkb= args[0]->val_str(&tmp_value);
|
|
|
|
|
|
|
|
if ((null_value=
|
|
|
|
args[0]->null_value ||
|
|
|
|
!(geom= Geometry::construct(&buffer, swkb->ptr(), swkb->length()))))
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
|
|
|
geom->get_mbr(&mbr, &c_end);
|
|
|
|
collector.set_extent(mbr.xmin, mbr.xmax, mbr.ymin, mbr.ymax);
|
|
|
|
if ((null_value= geom->store_shapes(&trn)))
|
|
|
|
{
|
|
|
|
str_value= 0;
|
|
|
|
goto mem_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
collector.prepare_operation();
|
|
|
|
if (!(cur_pi= collector.get_first()))
|
|
|
|
goto build_result; /* An EMPTY GEOMETRY */
|
|
|
|
|
|
|
|
if (!cur_pi->get_next())
|
|
|
|
{
|
|
|
|
/* Single point. */
|
2016-02-23 21:35:05 +01:00
|
|
|
if (res_receiver.single_point(cur_pi->node.shape.x, cur_pi->node.shape.y))
|
2014-11-27 21:29:37 +01:00
|
|
|
goto mem_error;
|
|
|
|
goto build_result;
|
|
|
|
}
|
|
|
|
|
|
|
|
left_cur= left_first= new_ch_node();
|
|
|
|
right_cur= right_first= new_ch_node();
|
|
|
|
right_first->prev= left_first->prev= 0;
|
|
|
|
right_first->pi= left_first->pi= cur_pi;
|
|
|
|
|
|
|
|
while ((cur_pi= cur_pi->get_next()))
|
|
|
|
{
|
|
|
|
/* Handle left part of the hull, then the right part. */
|
|
|
|
if (add_node_to_line(&left_cur, 1, cur_pi))
|
|
|
|
goto mem_error;
|
|
|
|
if (add_node_to_line(&right_cur, -1, cur_pi))
|
|
|
|
goto mem_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
left_cur->next= 0;
|
|
|
|
if (left_first->get_next()->get_next() == NULL &&
|
|
|
|
right_cur->prev->prev == NULL)
|
|
|
|
{
|
|
|
|
/* We only have 2 nodes in the result, so we create a polyline. */
|
|
|
|
if (res_receiver.start_shape(Gcalc_function::shape_line) ||
|
2016-02-23 21:35:05 +01:00
|
|
|
res_receiver.add_point(left_first->pi->node.shape.x, left_first->pi->node.shape.y) ||
|
|
|
|
res_receiver.add_point(left_cur->pi->node.shape.x, left_cur->pi->node.shape.y) ||
|
2014-11-27 21:29:37 +01:00
|
|
|
res_receiver.complete_shape())
|
|
|
|
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
goto build_result;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (res_receiver.start_shape(Gcalc_function::shape_polygon))
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
while (left_first)
|
|
|
|
{
|
2016-02-23 21:35:05 +01:00
|
|
|
if (res_receiver.add_point(left_first->pi->node.shape.x, left_first->pi->node.shape.y))
|
2014-11-27 21:29:37 +01:00
|
|
|
goto mem_error;
|
|
|
|
left_first= left_first->get_next();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Skip last point in the right part as it coincides */
|
|
|
|
/* with the last one in the left. */
|
|
|
|
right_cur= right_cur->prev;
|
|
|
|
while (right_cur->prev)
|
|
|
|
{
|
2016-02-23 21:35:05 +01:00
|
|
|
if (res_receiver.add_point(right_cur->pi->node.shape.x, right_cur->pi->node.shape.y))
|
2014-11-27 21:29:37 +01:00
|
|
|
goto mem_error;
|
|
|
|
right_cur= right_cur->prev;
|
|
|
|
}
|
|
|
|
res_receiver.complete_shape();
|
|
|
|
|
|
|
|
build_result:
|
|
|
|
str_value->set_charset(&my_charset_bin);
|
|
|
|
if (str_value->reserve(SRID_SIZE, 512))
|
|
|
|
goto mem_error;
|
|
|
|
str_value->length(0);
|
|
|
|
str_value->q_append(srid);
|
|
|
|
|
|
|
|
if (!Geometry::create_from_opresult(&buffer, str_value, res_receiver))
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
mem_error:
|
|
|
|
collector.reset();
|
|
|
|
func.reset();
|
|
|
|
res_receiver.reset();
|
|
|
|
res_heap.reset();
|
|
|
|
DBUG_RETURN(str_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
#else /*HEAVY_CONVEX_HULL*/
|
|
|
|
String *Item_func_convexhull::val_str(String *str_value)
|
|
|
|
{
|
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom= NULL;
|
|
|
|
MBR mbr;
|
|
|
|
const char *c_end;
|
|
|
|
Gcalc_operation_transporter trn(&func, &collector);
|
|
|
|
const Gcalc_scan_iterator::event_point *ev;
|
|
|
|
uint32 srid= 0;
|
|
|
|
ch_node *left_first, *left_cur, *right_first, *right_cur;
|
|
|
|
|
|
|
|
DBUG_ENTER("Item_func_convexhull::val_str");
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
|
|
|
String *swkb= args[0]->val_str(&tmp_value);
|
|
|
|
|
|
|
|
if ((null_value=
|
|
|
|
args[0]->null_value ||
|
|
|
|
!(geom= Geometry::construct(&buffer, swkb->ptr(), swkb->length()))))
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
|
|
|
geom->get_mbr(&mbr, &c_end);
|
|
|
|
collector.set_extent(mbr.xmin, mbr.xmax, mbr.ymin, mbr.ymax);
|
|
|
|
if ((null_value= geom->store_shapes(&trn)))
|
|
|
|
{
|
|
|
|
str_value= 0;
|
|
|
|
goto mem_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
collector.prepare_operation();
|
|
|
|
scan_it.init(&collector);
|
|
|
|
scan_it.killed= (int *) &(current_thd->killed);
|
|
|
|
|
|
|
|
if (!scan_it.more_points())
|
|
|
|
goto build_result; /* An EMPTY GEOMETRY */
|
|
|
|
|
|
|
|
if (scan_it.step())
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
if (!scan_it.more_points())
|
|
|
|
{
|
|
|
|
/* Single point. */
|
|
|
|
if (res_receiver.single_point(scan_it.get_events()->pi->x,
|
|
|
|
scan_it.get_events()->pi->y))
|
|
|
|
goto mem_error;
|
|
|
|
goto build_result;
|
|
|
|
}
|
|
|
|
|
|
|
|
left_cur= left_first= new_ch_node();
|
|
|
|
right_cur= right_first= new_ch_node();
|
|
|
|
right_first->prev= left_first->prev= 0;
|
|
|
|
right_first->pi= left_first->pi= scan_it.get_events()->pi;
|
|
|
|
|
|
|
|
while (scan_it.more_points())
|
|
|
|
{
|
|
|
|
if (scan_it.step())
|
|
|
|
goto mem_error;
|
|
|
|
ev= scan_it.get_events();
|
|
|
|
|
|
|
|
/* Skip the intersections-only events. */
|
|
|
|
while (ev->event == scev_intersection)
|
|
|
|
{
|
|
|
|
ev= ev->get_next();
|
|
|
|
if (!ev)
|
|
|
|
goto skip_point;
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
Gcalc_point_iterator pit(&scan_it);
|
|
|
|
if (!pit.point() || scan_it.get_event_position() == pit.point())
|
|
|
|
{
|
|
|
|
/* Handle left part of the hull. */
|
|
|
|
if (add_node_to_line(&left_cur, 1, ev->pi))
|
|
|
|
goto mem_error;
|
|
|
|
}
|
|
|
|
if (pit.point())
|
|
|
|
{
|
|
|
|
/* Check the rightmost point */
|
|
|
|
for(; pit.point()->c_get_next(); ++pit)
|
|
|
|
;
|
|
|
|
}
|
|
|
|
if (!pit.point() || pit.point()->event ||
|
|
|
|
scan_it.get_event_position() == pit.point()->c_get_next())
|
|
|
|
{
|
|
|
|
/* Handle right part of the hull. */
|
|
|
|
if (add_node_to_line(&right_cur, -1, ev->pi))
|
|
|
|
goto mem_error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
skip_point:;
|
|
|
|
}
|
|
|
|
|
|
|
|
left_cur->next= 0;
|
|
|
|
if (left_first->get_next()->get_next() == NULL &&
|
|
|
|
right_cur->prev->prev == NULL)
|
|
|
|
{
|
|
|
|
/* We only have 2 nodes in the result, so we create a polyline. */
|
|
|
|
if (res_receiver.start_shape(Gcalc_function::shape_line) ||
|
|
|
|
res_receiver.add_point(left_first->pi->x, left_first->pi->y) ||
|
|
|
|
res_receiver.add_point(left_cur->pi->x, left_cur->pi->y) ||
|
|
|
|
res_receiver.complete_shape())
|
|
|
|
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
goto build_result;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (res_receiver.start_shape(Gcalc_function::shape_polygon))
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
while (left_first)
|
|
|
|
{
|
|
|
|
if (res_receiver.add_point(left_first->pi->x, left_first->pi->y))
|
|
|
|
goto mem_error;
|
|
|
|
left_first= left_first->get_next();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Skip last point in the right part as it coincides */
|
|
|
|
/* with the last one in the left. */
|
|
|
|
right_cur= right_cur->prev;
|
|
|
|
while (right_cur->prev)
|
|
|
|
{
|
|
|
|
if (res_receiver.add_point(right_cur->pi->x, right_cur->pi->y))
|
|
|
|
goto mem_error;
|
|
|
|
right_cur= right_cur->prev;
|
|
|
|
}
|
|
|
|
res_receiver.complete_shape();
|
|
|
|
|
|
|
|
build_result:
|
|
|
|
str_value->set_charset(&my_charset_bin);
|
|
|
|
if (str_value->reserve(SRID_SIZE, 512))
|
|
|
|
goto mem_error;
|
|
|
|
str_value->length(0);
|
|
|
|
str_value->q_append(srid);
|
|
|
|
|
|
|
|
if (!Geometry::create_from_opresult(&buffer, str_value, res_receiver))
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
mem_error:
|
|
|
|
collector.reset();
|
|
|
|
func.reset();
|
|
|
|
res_receiver.reset();
|
|
|
|
res_heap.reset();
|
|
|
|
DBUG_RETURN(str_value);
|
|
|
|
}
|
|
|
|
#endif /*HEAVY_CONVEX_HULL*/
|
|
|
|
|
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
/*
|
|
|
|
Spatial decomposition functions
|
|
|
|
*/
|
|
|
|
|
|
|
|
String *Item_func_spatial_decomp::val_str(String *str)
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-30 12:22:34 +02:00
|
|
|
String arg_val;
|
|
|
|
String *swkb= args[0]->val_str(&arg_val);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom= NULL;
|
2004-03-04 07:50:37 +01:00
|
|
|
uint32 srid;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-03-12 09:04:00 +01:00
|
|
|
if ((null_value=
|
|
|
|
(args[0]->null_value ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer, swkb->ptr(), swkb->length())))))
|
2003-05-30 12:22:34 +02:00
|
|
|
return 0;
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
srid= uint4korr(swkb->ptr());
|
2004-09-07 13:44:25 +02:00
|
|
|
str->set_charset(&my_charset_bin);
|
2003-05-30 12:22:34 +02:00
|
|
|
if (str->reserve(SRID_SIZE, 512))
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
str->length(0);
|
|
|
|
str->q_append(srid);
|
2004-03-04 07:50:37 +01:00
|
|
|
switch (decomp_func) {
|
2003-05-30 12:22:34 +02:00
|
|
|
case SP_STARTPOINT:
|
2004-03-12 09:04:00 +01:00
|
|
|
if (geom->start_point(str))
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SP_ENDPOINT:
|
2004-03-12 09:04:00 +01:00
|
|
|
if (geom->end_point(str))
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SP_EXTERIORRING:
|
2004-03-12 09:04:00 +01:00
|
|
|
if (geom->exterior_ring(str))
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
2004-03-04 07:50:37 +01:00
|
|
|
return str;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
err:
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_spatial_decomp_n::val_str(String *str)
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-30 12:22:34 +02:00
|
|
|
String arg_val;
|
|
|
|
String *swkb= args[0]->val_str(&arg_val);
|
|
|
|
long n= (long) args[1]->val_int();
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom= NULL;
|
2004-03-04 07:50:37 +01:00
|
|
|
uint32 srid;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-03-12 09:04:00 +01:00
|
|
|
if ((null_value=
|
|
|
|
(args[0]->null_value || args[1]->null_value ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer, swkb->ptr(), swkb->length())))))
|
2003-05-30 12:22:34 +02:00
|
|
|
return 0;
|
|
|
|
|
2004-09-07 13:44:25 +02:00
|
|
|
str->set_charset(&my_charset_bin);
|
2003-05-30 12:22:34 +02:00
|
|
|
if (str->reserve(SRID_SIZE, 512))
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
|
|
|
srid= uint4korr(swkb->ptr());
|
2003-05-30 12:22:34 +02:00
|
|
|
str->length(0);
|
|
|
|
str->q_append(srid);
|
2004-03-04 07:50:37 +01:00
|
|
|
switch (decomp_func_n)
|
2003-05-30 12:22:34 +02:00
|
|
|
{
|
|
|
|
case SP_POINTN:
|
2004-03-12 09:04:00 +01:00
|
|
|
if (geom->point_n(n,str))
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SP_GEOMETRYN:
|
2004-03-12 09:04:00 +01:00
|
|
|
if (geom->geometry_n(n,str))
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SP_INTERIORRINGN:
|
2004-03-12 09:04:00 +01:00
|
|
|
if (geom->interior_ring_n(n,str))
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
2004-03-04 07:50:37 +01:00
|
|
|
return str;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
err:
|
|
|
|
null_value=1;
|
|
|
|
return 0;
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2004-07-02 23:22:46 +02:00
|
|
|
Functions to concatenate various spatial objects
|
2003-05-30 12:22:34 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2004-07-02 23:22:46 +02:00
|
|
|
* Concatenate doubles into Point
|
2003-05-30 12:22:34 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_point::val_str(String *str)
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2004-11-11 19:39:35 +01:00
|
|
|
double x= args[0]->val_real();
|
|
|
|
double y= args[1]->val_real();
|
2009-04-28 11:47:26 +02:00
|
|
|
uint32 srid= 0;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
if ((null_value= (args[0]->null_value ||
|
2011-05-04 20:20:17 +02:00
|
|
|
args[1]->null_value ||
|
2018-07-12 17:12:20 +02:00
|
|
|
str->alloc(4/*SRID*/ + 1 + 4 + SIZEOF_STORED_DOUBLE * 2))))
|
2003-05-30 12:22:34 +02:00
|
|
|
return 0;
|
|
|
|
|
2004-09-07 13:44:25 +02:00
|
|
|
str->set_charset(&my_charset_bin);
|
2003-05-30 12:22:34 +02:00
|
|
|
str->length(0);
|
2009-04-28 11:47:26 +02:00
|
|
|
str->q_append(srid);
|
2004-03-12 09:04:00 +01:00
|
|
|
str->q_append((char)Geometry::wkb_ndr);
|
|
|
|
str->q_append((uint32)Geometry::wkb_point);
|
2003-05-30 12:22:34 +02:00
|
|
|
str->q_append(x);
|
|
|
|
str->q_append(y);
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
/**
|
2004-07-02 23:22:46 +02:00
|
|
|
Concatenates various items into various collections
|
2003-05-30 12:22:34 +02:00
|
|
|
with checkings for valid wkb type of items.
|
|
|
|
For example, MultiPoint can be a collection of Points only.
|
|
|
|
coll_type contains wkb type of target collection.
|
|
|
|
item_type contains a valid wkb type of items.
|
|
|
|
In the case when coll_type is wkbGeometryCollection,
|
|
|
|
we do not check wkb type of items, any is valid.
|
|
|
|
*/
|
|
|
|
|
|
|
|
String *Item_func_spatial_collection::val_str(String *str)
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-30 12:22:34 +02:00
|
|
|
String arg_value;
|
|
|
|
uint i;
|
2009-04-28 11:47:26 +02:00
|
|
|
uint32 srid= 0;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-09-07 13:44:25 +02:00
|
|
|
str->set_charset(&my_charset_bin);
|
2003-05-30 12:22:34 +02:00
|
|
|
str->length(0);
|
2009-04-28 11:47:26 +02:00
|
|
|
if (str->reserve(4/*SRID*/ + 1 + 4 + 4, 512))
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2009-04-28 11:47:26 +02:00
|
|
|
str->q_append(srid);
|
2004-03-12 09:04:00 +01:00
|
|
|
str->q_append((char) Geometry::wkb_ndr);
|
2003-05-30 12:22:34 +02:00
|
|
|
str->q_append((uint32) coll_type);
|
|
|
|
str->q_append((uint32) arg_count);
|
|
|
|
|
|
|
|
for (i= 0; i < arg_count; ++i)
|
|
|
|
{
|
|
|
|
String *res= args[i]->val_str(&arg_value);
|
2007-02-21 11:45:19 +01:00
|
|
|
uint32 len;
|
|
|
|
if (args[i]->null_value || ((len= res->length()) < WKB_HEADER_SIZE))
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-03-12 09:04:00 +01:00
|
|
|
if (coll_type == Geometry::wkb_geometrycollection)
|
2003-05-30 12:22:34 +02:00
|
|
|
{
|
|
|
|
/*
|
2004-03-04 07:50:37 +01:00
|
|
|
In the case of GeometryCollection we don't need any checkings
|
|
|
|
for item types, so just copy them into target collection
|
2003-05-30 12:22:34 +02:00
|
|
|
*/
|
2009-04-28 11:47:26 +02:00
|
|
|
if (str->append(res->ptr() + 4/*SRID*/, len - 4/*SRID*/, (uint32) 512))
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
enum Geometry::wkbType wkb_type;
|
2009-06-17 16:58:33 +02:00
|
|
|
const uint data_offset= 4/*SRID*/ + 1;
|
|
|
|
if (res->length() < data_offset + sizeof(uint32))
|
|
|
|
goto err;
|
|
|
|
const char *data= res->ptr() + data_offset;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
/*
|
2004-07-02 23:22:46 +02:00
|
|
|
In the case of named collection we must check that items
|
2004-03-04 07:50:37 +01:00
|
|
|
are of specific type, let's do this checking now
|
2003-05-30 12:22:34 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
wkb_type= (Geometry::wkbType) uint4korr(data);
|
|
|
|
data+= 4;
|
2009-04-28 11:47:26 +02:00
|
|
|
len-= 5 + 4/*SRID*/;
|
2003-05-30 12:22:34 +02:00
|
|
|
if (wkb_type != item_type)
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
switch (coll_type) {
|
2004-03-12 09:04:00 +01:00
|
|
|
case Geometry::wkb_multipoint:
|
|
|
|
case Geometry::wkb_multilinestring:
|
|
|
|
case Geometry::wkb_multipolygon:
|
2004-03-04 07:50:37 +01:00
|
|
|
if (len < WKB_HEADER_SIZE ||
|
|
|
|
str->append(data-WKB_HEADER_SIZE, len+WKB_HEADER_SIZE, 512))
|
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
break;
|
|
|
|
|
2004-03-12 09:04:00 +01:00
|
|
|
case Geometry::wkb_linestring:
|
2009-06-17 16:58:33 +02:00
|
|
|
if (len < POINT_DATA_SIZE || str->append(data, POINT_DATA_SIZE, 512))
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
break;
|
2004-03-12 09:04:00 +01:00
|
|
|
case Geometry::wkb_polygon:
|
2003-05-30 12:22:34 +02:00
|
|
|
{
|
|
|
|
uint32 n_points;
|
|
|
|
double x1, y1, x2, y2;
|
2004-03-04 07:50:37 +01:00
|
|
|
const char *org_data= data;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2009-06-17 16:58:33 +02:00
|
|
|
if (len < 4)
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
n_points= uint4korr(data);
|
|
|
|
data+= 4;
|
2009-06-17 16:58:33 +02:00
|
|
|
|
|
|
|
if (n_points < 2 || len < 4 + n_points * POINT_DATA_SIZE)
|
|
|
|
goto err;
|
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
float8get(x1, data);
|
2004-03-04 07:50:37 +01:00
|
|
|
data+= SIZEOF_STORED_DOUBLE;
|
2003-05-30 12:22:34 +02:00
|
|
|
float8get(y1, data);
|
2004-03-04 07:50:37 +01:00
|
|
|
data+= SIZEOF_STORED_DOUBLE;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
data+= (n_points - 2) * POINT_DATA_SIZE;
|
|
|
|
|
|
|
|
float8get(x2, data);
|
2004-03-04 07:50:37 +01:00
|
|
|
float8get(y2, data + SIZEOF_STORED_DOUBLE);
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
if ((x1 != x2) || (y1 != y2) ||
|
|
|
|
str->append(org_data, len, 512))
|
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (str->length() > current_thd->variables.max_allowed_packet)
|
2004-06-29 16:55:13 +02:00
|
|
|
{
|
2015-07-06 19:24:14 +02:00
|
|
|
THD *thd= current_thd;
|
|
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
2004-06-29 16:55:13 +02:00
|
|
|
ER_WARN_ALLOWED_PACKET_OVERFLOWED,
|
2015-07-06 19:24:14 +02:00
|
|
|
ER_THD(thd, ER_WARN_ALLOWED_PACKET_OVERFLOWED),
|
|
|
|
func_name(), thd->variables.max_allowed_packet);
|
2004-03-04 07:50:37 +01:00
|
|
|
goto err;
|
2004-06-29 16:55:13 +02:00
|
|
|
}
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
null_value = 0;
|
2004-03-04 07:50:37 +01:00
|
|
|
return str;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
err:
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
/*
|
|
|
|
Functions for spatial relations
|
|
|
|
*/
|
|
|
|
|
2015-08-13 12:25:51 +02:00
|
|
|
static SEL_ARG sel_arg_impossible(SEL_ARG::IMPOSSIBLE);
|
|
|
|
|
|
|
|
SEL_ARG *
|
|
|
|
Item_func_spatial_rel::get_mm_leaf(RANGE_OPT_PARAM *param,
|
|
|
|
Field *field, KEY_PART *key_part,
|
|
|
|
Item_func::Functype type, Item *value)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_func_spatial_rel::get_mm_leaf");
|
|
|
|
if (key_part->image_type != Field::itMBR)
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
if (value->cmp_type() != STRING_RESULT)
|
|
|
|
DBUG_RETURN(&sel_arg_impossible);
|
|
|
|
|
|
|
|
if (param->using_real_indexes &&
|
|
|
|
!field->optimize_range(param->real_keynr[key_part->key],
|
|
|
|
key_part->part))
|
2019-07-05 18:52:59 +02:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
|
|
|
Field_geom *field_geom= dynamic_cast<Field_geom*>(field);
|
|
|
|
DBUG_ASSERT(field_geom);
|
|
|
|
const Type_handler_geometry *sav_geom_type= field_geom->type_handler_geom();
|
|
|
|
// We have to be able to store all sorts of spatial features here
|
|
|
|
field_geom->set_type_handler(&type_handler_geometry);
|
|
|
|
bool rc= value->save_in_field_no_warnings(field, 1);
|
|
|
|
field_geom->set_type_handler(sav_geom_type);
|
2015-08-13 12:25:51 +02:00
|
|
|
|
2019-07-05 18:52:59 +02:00
|
|
|
if (rc)
|
2015-08-13 12:25:51 +02:00
|
|
|
DBUG_RETURN(&sel_arg_impossible); // Bad GEOMETRY value
|
|
|
|
|
|
|
|
DBUG_ASSERT(!field->real_maybe_null()); // SPATIAL keys do not support NULL
|
|
|
|
|
|
|
|
uchar *str= (uchar*) alloc_root(param->mem_root, key_part->store_length + 1);
|
|
|
|
if (!str)
|
|
|
|
DBUG_RETURN(0); // out of memory
|
|
|
|
field->get_key_image(str, key_part->length, key_part->image_type);
|
|
|
|
SEL_ARG *tree;
|
|
|
|
if (!(tree= new (param->mem_root) SEL_ARG(field, str, str)))
|
|
|
|
DBUG_RETURN(0); // out of memory
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case SP_EQUALS_FUNC:
|
|
|
|
tree->min_flag= GEOM_FLAG | HA_READ_MBR_EQUAL;// NEAR_MIN;//512;
|
|
|
|
tree->max_flag= NO_MAX_RANGE;
|
|
|
|
break;
|
|
|
|
case SP_DISJOINT_FUNC:
|
|
|
|
tree->min_flag= GEOM_FLAG | HA_READ_MBR_DISJOINT;// NEAR_MIN;//512;
|
|
|
|
tree->max_flag= NO_MAX_RANGE;
|
|
|
|
break;
|
|
|
|
case SP_INTERSECTS_FUNC:
|
|
|
|
tree->min_flag= GEOM_FLAG | HA_READ_MBR_INTERSECT;// NEAR_MIN;//512;
|
|
|
|
tree->max_flag= NO_MAX_RANGE;
|
|
|
|
break;
|
|
|
|
case SP_TOUCHES_FUNC:
|
|
|
|
tree->min_flag= GEOM_FLAG | HA_READ_MBR_INTERSECT;// NEAR_MIN;//512;
|
|
|
|
tree->max_flag= NO_MAX_RANGE;
|
|
|
|
break;
|
|
|
|
case SP_CROSSES_FUNC:
|
|
|
|
tree->min_flag= GEOM_FLAG | HA_READ_MBR_INTERSECT;// NEAR_MIN;//512;
|
|
|
|
tree->max_flag= NO_MAX_RANGE;
|
|
|
|
break;
|
|
|
|
case SP_WITHIN_FUNC:
|
2017-05-02 14:24:42 +02:00
|
|
|
tree->min_flag= GEOM_FLAG | HA_READ_MBR_CONTAIN;// NEAR_MIN;//512;
|
2015-08-13 12:25:51 +02:00
|
|
|
tree->max_flag= NO_MAX_RANGE;
|
|
|
|
break;
|
|
|
|
case SP_CONTAINS_FUNC:
|
2017-05-02 14:24:42 +02:00
|
|
|
tree->min_flag= GEOM_FLAG | HA_READ_MBR_WITHIN;// NEAR_MIN;//512;
|
2015-08-13 12:25:51 +02:00
|
|
|
tree->max_flag= NO_MAX_RANGE;
|
|
|
|
break;
|
|
|
|
case SP_OVERLAPS_FUNC:
|
|
|
|
tree->min_flag= GEOM_FLAG | HA_READ_MBR_INTERSECT;// NEAR_MIN;//512;
|
|
|
|
tree->max_flag= NO_MAX_RANGE;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
DBUG_ASSERT(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
DBUG_RETURN(tree);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
const char *Item_func_spatial_mbr_rel::func_name() const
|
|
|
|
{
|
|
|
|
switch (spatial_rel) {
|
|
|
|
case SP_CONTAINS_FUNC:
|
|
|
|
return "mbrcontains";
|
|
|
|
case SP_WITHIN_FUNC:
|
|
|
|
return "mbrwithin";
|
|
|
|
case SP_EQUALS_FUNC:
|
|
|
|
return "mbrequals";
|
|
|
|
case SP_DISJOINT_FUNC:
|
|
|
|
return "mbrdisjoint";
|
|
|
|
case SP_INTERSECTS_FUNC:
|
|
|
|
return "mbrintersects";
|
|
|
|
case SP_TOUCHES_FUNC:
|
|
|
|
return "mbrtouches";
|
|
|
|
case SP_CROSSES_FUNC:
|
|
|
|
return "mbrcrosses";
|
|
|
|
case SP_OVERLAPS_FUNC:
|
|
|
|
return "mbroverlaps";
|
|
|
|
default:
|
|
|
|
DBUG_ASSERT(0); // Should never happened
|
|
|
|
return "mbrsp_unknown";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
longlong Item_func_spatial_mbr_rel::val_int()
|
2003-05-30 12:22:34 +02:00
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2015-05-05 07:30:17 +02:00
|
|
|
String *res1= args[0]->val_str(&tmp_value1);
|
|
|
|
String *res2= args[1]->val_str(&tmp_value2);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer1, buffer2;
|
|
|
|
Geometry *g1, *g2;
|
2003-05-30 12:22:34 +02:00
|
|
|
MBR mbr1, mbr2;
|
2004-03-04 07:50:37 +01:00
|
|
|
const char *dummy;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2004-03-12 09:04:00 +01:00
|
|
|
if ((null_value=
|
|
|
|
(args[0]->null_value ||
|
|
|
|
args[1]->null_value ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(g1= Geometry::construct(&buffer1, res1->ptr(), res1->length())) ||
|
|
|
|
!(g2= Geometry::construct(&buffer2, res2->ptr(), res2->length())) ||
|
2013-03-26 18:47:06 +01:00
|
|
|
g1->get_mbr(&mbr1, &dummy) || !mbr1.valid() ||
|
|
|
|
g2->get_mbr(&mbr2, &dummy) || !mbr2.valid())))
|
2003-05-30 12:22:34 +02:00
|
|
|
return 0;
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
switch (spatial_rel) {
|
2003-05-30 12:22:34 +02:00
|
|
|
case SP_CONTAINS_FUNC:
|
|
|
|
return mbr1.contains(&mbr2);
|
|
|
|
case SP_WITHIN_FUNC:
|
|
|
|
return mbr1.within(&mbr2);
|
|
|
|
case SP_EQUALS_FUNC:
|
|
|
|
return mbr1.equals(&mbr2);
|
|
|
|
case SP_DISJOINT_FUNC:
|
|
|
|
return mbr1.disjoint(&mbr2);
|
|
|
|
case SP_INTERSECTS_FUNC:
|
|
|
|
return mbr1.intersects(&mbr2);
|
|
|
|
case SP_TOUCHES_FUNC:
|
|
|
|
return mbr1.touches(&mbr2);
|
|
|
|
case SP_OVERLAPS_FUNC:
|
|
|
|
return mbr1.overlaps(&mbr2);
|
|
|
|
case SP_CROSSES_FUNC:
|
|
|
|
return 0;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
null_value=1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2015-05-05 07:30:17 +02:00
|
|
|
const char *Item_func_spatial_precise_rel::func_name() const
|
2011-05-04 20:20:17 +02:00
|
|
|
{
|
|
|
|
switch (spatial_rel) {
|
|
|
|
case SP_CONTAINS_FUNC:
|
|
|
|
return "st_contains";
|
|
|
|
case SP_WITHIN_FUNC:
|
|
|
|
return "st_within";
|
|
|
|
case SP_EQUALS_FUNC:
|
|
|
|
return "st_equals";
|
|
|
|
case SP_DISJOINT_FUNC:
|
|
|
|
return "st_disjoint";
|
|
|
|
case SP_INTERSECTS_FUNC:
|
|
|
|
return "st_intersects";
|
|
|
|
case SP_TOUCHES_FUNC:
|
|
|
|
return "st_touches";
|
|
|
|
case SP_CROSSES_FUNC:
|
|
|
|
return "st_crosses";
|
|
|
|
case SP_OVERLAPS_FUNC:
|
|
|
|
return "st_overlaps";
|
|
|
|
default:
|
|
|
|
DBUG_ASSERT(0); // Should never happened
|
|
|
|
return "sp_unknown";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static double count_edge_t(const Gcalc_heap::Info *ea,
|
|
|
|
const Gcalc_heap::Info *eb,
|
|
|
|
const Gcalc_heap::Info *v,
|
|
|
|
double &ex, double &ey, double &vx, double &vy,
|
|
|
|
double &e_sqrlen)
|
|
|
|
{
|
2016-02-14 20:57:48 +01:00
|
|
|
ex= eb->node.shape.x - ea->node.shape.x;
|
|
|
|
ey= eb->node.shape.y - ea->node.shape.y;
|
|
|
|
vx= v->node.shape.x - ea->node.shape.x;
|
|
|
|
vy= v->node.shape.y - ea->node.shape.y;
|
2011-05-04 20:20:17 +02:00
|
|
|
e_sqrlen= ex * ex + ey * ey;
|
|
|
|
return (ex * vx + ey * vy) / e_sqrlen;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static double distance_to_line(double ex, double ey, double vx, double vy,
|
|
|
|
double e_sqrlen)
|
|
|
|
{
|
|
|
|
return fabs(vx * ey - vy * ex) / sqrt(e_sqrlen);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static double distance_points(const Gcalc_heap::Info *a,
|
|
|
|
const Gcalc_heap::Info *b)
|
|
|
|
{
|
2016-02-14 20:57:48 +01:00
|
|
|
double x= a->node.shape.x - b->node.shape.x;
|
|
|
|
double y= a->node.shape.y - b->node.shape.y;
|
2011-05-04 20:20:17 +02:00
|
|
|
return sqrt(x * x + y * y);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-11-27 21:29:37 +01:00
|
|
|
static Gcalc_function::op_type op_matrix(int n)
|
|
|
|
{
|
|
|
|
switch (n)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
return Gcalc_function::op_internals;
|
2015-03-15 19:20:38 +01:00
|
|
|
case 1:
|
|
|
|
return Gcalc_function::op_border;
|
2014-11-27 21:29:37 +01:00
|
|
|
case 2:
|
|
|
|
return (Gcalc_function::op_type)
|
|
|
|
((int) Gcalc_function::op_not | (int) Gcalc_function::op_union);
|
|
|
|
};
|
|
|
|
GCALC_DBUG_ASSERT(FALSE);
|
|
|
|
return Gcalc_function::op_any;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int setup_relate_func(Geometry *g1, Geometry *g2,
|
|
|
|
Gcalc_operation_transporter *trn, Gcalc_function *func,
|
|
|
|
const char *mask)
|
|
|
|
{
|
|
|
|
int do_store_shapes=1;
|
2015-08-25 09:46:31 +02:00
|
|
|
uint UNINIT_VAR(shape_a), UNINIT_VAR(shape_b);
|
2014-11-27 21:29:37 +01:00
|
|
|
uint n_operands= 0;
|
|
|
|
int last_shape_pos;
|
|
|
|
|
|
|
|
last_shape_pos= func->get_next_expression_pos();
|
2015-03-15 19:20:38 +01:00
|
|
|
if (func->reserve_op_buffer(1))
|
|
|
|
return 1;
|
2014-11-27 21:29:37 +01:00
|
|
|
func->add_operation(Gcalc_function::op_intersection, 0);
|
|
|
|
for (int nc=0; nc<9; nc++)
|
|
|
|
{
|
|
|
|
uint cur_op;
|
|
|
|
|
|
|
|
cur_op= Gcalc_function::op_intersection;
|
|
|
|
switch (mask[nc])
|
|
|
|
{
|
|
|
|
case '*':
|
|
|
|
continue;
|
|
|
|
case 'T':
|
|
|
|
case '0':
|
|
|
|
case '1':
|
|
|
|
case '2':
|
|
|
|
cur_op|= Gcalc_function::v_find_t;
|
|
|
|
break;
|
|
|
|
case 'F':
|
2015-06-22 20:33:35 +02:00
|
|
|
cur_op|= (Gcalc_function::op_not | Gcalc_function::v_find_f);
|
2014-11-27 21:29:37 +01:00
|
|
|
break;
|
2015-06-22 20:33:35 +02:00
|
|
|
default:
|
|
|
|
return 1;
|
2014-11-27 21:29:37 +01:00
|
|
|
};
|
|
|
|
++n_operands;
|
2015-03-15 19:20:38 +01:00
|
|
|
if (func->reserve_op_buffer(3))
|
2014-11-27 21:29:37 +01:00
|
|
|
return 1;
|
|
|
|
func->add_operation(cur_op, 2);
|
|
|
|
|
|
|
|
func->add_operation(op_matrix(nc/3), 1);
|
|
|
|
if (do_store_shapes)
|
|
|
|
{
|
|
|
|
shape_a= func->get_next_expression_pos();
|
|
|
|
if (g1->store_shapes(trn))
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
func->repeat_expression(shape_a);
|
2017-10-07 12:17:45 +02:00
|
|
|
if (func->reserve_op_buffer(1))
|
|
|
|
return 1;
|
2014-11-27 21:29:37 +01:00
|
|
|
func->add_operation(op_matrix(nc%3), 1);
|
|
|
|
if (do_store_shapes)
|
|
|
|
{
|
|
|
|
shape_b= func->get_next_expression_pos();
|
|
|
|
if (g2->store_shapes(trn))
|
|
|
|
return 1;
|
|
|
|
do_store_shapes= 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
func->repeat_expression(shape_b);
|
|
|
|
}
|
|
|
|
|
|
|
|
func->add_operands_to_op(last_shape_pos, n_operands);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
#define GIS_ZERO 0.00000000001
|
|
|
|
|
2015-06-26 13:42:49 +02:00
|
|
|
class Geometry_ptr_with_buffer_and_mbr
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Geometry *geom;
|
|
|
|
Geometry_buffer buffer;
|
|
|
|
MBR mbr;
|
|
|
|
bool construct(Item *item, String *tmp_value)
|
|
|
|
{
|
|
|
|
const char *c_end;
|
|
|
|
String *res= item->val_str(tmp_value);
|
|
|
|
return
|
|
|
|
item->null_value ||
|
|
|
|
!(geom= Geometry::construct(&buffer, res->ptr(), res->length())) ||
|
|
|
|
geom->get_mbr(&mbr, &c_end) || !mbr.valid();
|
|
|
|
}
|
|
|
|
int store_shapes(Gcalc_shape_transporter *trn) const
|
|
|
|
{ return geom->store_shapes(trn); }
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
longlong Item_func_spatial_relate::val_int()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_func_spatial_relate::val_int");
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
|
|
|
Geometry_ptr_with_buffer_and_mbr g1, g2;
|
|
|
|
int result= 0;
|
|
|
|
|
|
|
|
if ((null_value= (g1.construct(args[0], &tmp_value1) ||
|
|
|
|
g2.construct(args[1], &tmp_value2) ||
|
|
|
|
func.reserve_op_buffer(1))))
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
|
|
|
MBR umbr(g1.mbr, g2.mbr);
|
|
|
|
collector.set_extent(umbr.xmin, umbr.xmax, umbr.ymin, umbr.ymax);
|
|
|
|
g1.mbr.buffer(1e-5);
|
|
|
|
Gcalc_operation_transporter trn(&func, &collector);
|
|
|
|
|
|
|
|
String *matrix= args[2]->val_str(&tmp_matrix);
|
|
|
|
if ((null_value= args[2]->null_value || matrix->length() != 9 ||
|
|
|
|
setup_relate_func(g1.geom, g2.geom,
|
|
|
|
&trn, &func, matrix->ptr())))
|
|
|
|
goto exit;
|
|
|
|
|
|
|
|
collector.prepare_operation();
|
|
|
|
scan_it.init(&collector);
|
|
|
|
scan_it.killed= (int *) &(current_thd->killed);
|
|
|
|
if (!func.alloc_states())
|
|
|
|
result= func.check_function(scan_it);
|
|
|
|
|
|
|
|
exit:
|
|
|
|
collector.reset();
|
|
|
|
func.reset();
|
|
|
|
scan_it.reset();
|
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-05-05 07:30:17 +02:00
|
|
|
longlong Item_func_spatial_precise_rel::val_int()
|
2011-05-04 20:20:17 +02:00
|
|
|
{
|
2015-05-05 07:30:17 +02:00
|
|
|
DBUG_ENTER("Item_func_spatial_precise_rel::val_int");
|
2011-05-04 20:20:17 +02:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2015-06-26 13:42:49 +02:00
|
|
|
Geometry_ptr_with_buffer_and_mbr g1, g2;
|
2011-05-04 20:20:17 +02:00
|
|
|
int result= 0;
|
2011-09-22 15:53:36 +02:00
|
|
|
uint shape_a, shape_b;
|
2011-05-04 20:20:17 +02:00
|
|
|
|
2015-06-26 13:42:49 +02:00
|
|
|
if ((null_value= (g1.construct(args[0], &tmp_value1) ||
|
|
|
|
g2.construct(args[1], &tmp_value2) ||
|
|
|
|
func.reserve_op_buffer(1))))
|
2011-05-04 20:20:17 +02:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
2015-06-26 13:42:49 +02:00
|
|
|
Gcalc_operation_transporter trn(&func, &collector);
|
2011-09-22 15:53:36 +02:00
|
|
|
|
2015-06-26 13:42:49 +02:00
|
|
|
MBR umbr(g1.mbr, g2.mbr);
|
2011-12-08 13:29:45 +01:00
|
|
|
collector.set_extent(umbr.xmin, umbr.xmax, umbr.ymin, umbr.ymax);
|
|
|
|
|
2015-06-26 13:42:49 +02:00
|
|
|
g1.mbr.buffer(1e-5);
|
2011-12-08 13:29:45 +01:00
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
switch (spatial_rel) {
|
|
|
|
case SP_CONTAINS_FUNC:
|
2015-06-26 13:42:49 +02:00
|
|
|
if (!g1.mbr.contains(&g2.mbr))
|
2011-12-08 13:29:45 +01:00
|
|
|
goto exit;
|
2015-06-22 20:33:35 +02:00
|
|
|
func.add_operation(Gcalc_function::v_find_f |
|
|
|
|
Gcalc_function::op_not |
|
|
|
|
Gcalc_function::op_difference, 2);
|
2011-09-22 15:53:36 +02:00
|
|
|
/* Mind the g2 goes first. */
|
2015-06-26 13:42:49 +02:00
|
|
|
null_value= g2.store_shapes(&trn) || g1.store_shapes(&trn);
|
2011-05-04 20:20:17 +02:00
|
|
|
break;
|
|
|
|
case SP_WITHIN_FUNC:
|
2015-06-26 13:42:49 +02:00
|
|
|
g2.mbr.buffer(2e-5);
|
|
|
|
if (!g1.mbr.within(&g2.mbr))
|
2011-12-08 13:29:45 +01:00
|
|
|
goto exit;
|
2015-06-22 20:33:35 +02:00
|
|
|
func.add_operation(Gcalc_function::v_find_f |
|
|
|
|
Gcalc_function::op_not |
|
|
|
|
Gcalc_function::op_difference, 2);
|
2015-06-26 13:42:49 +02:00
|
|
|
null_value= g1.store_shapes(&trn) || g2.store_shapes(&trn);
|
2011-05-04 20:20:17 +02:00
|
|
|
break;
|
|
|
|
case SP_EQUALS_FUNC:
|
2015-06-26 13:42:49 +02:00
|
|
|
if (!g1.mbr.contains(&g2.mbr))
|
2011-12-08 13:29:45 +01:00
|
|
|
goto exit;
|
2015-06-22 20:33:35 +02:00
|
|
|
func.add_operation(Gcalc_function::v_find_f |
|
|
|
|
Gcalc_function::op_not |
|
|
|
|
Gcalc_function::op_symdifference, 2);
|
2015-06-26 13:42:49 +02:00
|
|
|
null_value= g1.store_shapes(&trn) || g2.store_shapes(&trn);
|
2011-05-04 20:20:17 +02:00
|
|
|
break;
|
|
|
|
case SP_DISJOINT_FUNC:
|
2015-06-22 20:33:35 +02:00
|
|
|
func.add_operation(Gcalc_function::v_find_f |
|
|
|
|
Gcalc_function::op_not |
|
|
|
|
Gcalc_function::op_intersection, 2);
|
2015-06-26 13:42:49 +02:00
|
|
|
null_value= g1.store_shapes(&trn) || g2.store_shapes(&trn);
|
2011-05-04 20:20:17 +02:00
|
|
|
break;
|
|
|
|
case SP_INTERSECTS_FUNC:
|
2015-06-26 13:42:49 +02:00
|
|
|
if (!g1.mbr.intersects(&g2.mbr))
|
2011-12-08 13:29:45 +01:00
|
|
|
goto exit;
|
2015-06-22 20:33:35 +02:00
|
|
|
func.add_operation(Gcalc_function::v_find_t |
|
|
|
|
Gcalc_function::op_intersection, 2);
|
2015-06-26 13:42:49 +02:00
|
|
|
null_value= g1.store_shapes(&trn) || g2.store_shapes(&trn);
|
2011-05-04 20:20:17 +02:00
|
|
|
break;
|
|
|
|
case SP_OVERLAPS_FUNC:
|
|
|
|
case SP_CROSSES_FUNC:
|
|
|
|
func.add_operation(Gcalc_function::op_intersection, 2);
|
2015-06-22 20:33:35 +02:00
|
|
|
if (func.reserve_op_buffer(3))
|
2014-11-27 21:29:37 +01:00
|
|
|
break;
|
2011-09-22 15:53:36 +02:00
|
|
|
func.add_operation(Gcalc_function::v_find_t |
|
|
|
|
Gcalc_function::op_intersection, 2);
|
|
|
|
shape_a= func.get_next_expression_pos();
|
2015-06-26 13:42:49 +02:00
|
|
|
if ((null_value= g1.store_shapes(&trn)))
|
2011-09-22 15:53:36 +02:00
|
|
|
break;
|
|
|
|
shape_b= func.get_next_expression_pos();
|
2015-06-26 13:42:49 +02:00
|
|
|
if ((null_value= g2.store_shapes(&trn)))
|
2011-09-22 15:53:36 +02:00
|
|
|
break;
|
2014-11-27 21:29:37 +01:00
|
|
|
if (func.reserve_op_buffer(7))
|
|
|
|
break;
|
2015-06-22 20:33:35 +02:00
|
|
|
func.add_operation(Gcalc_function::op_intersection, 2);
|
2011-09-22 15:53:36 +02:00
|
|
|
func.add_operation(Gcalc_function::v_find_t |
|
|
|
|
Gcalc_function::op_difference, 2);
|
|
|
|
func.repeat_expression(shape_a);
|
|
|
|
func.repeat_expression(shape_b);
|
|
|
|
func.add_operation(Gcalc_function::v_find_t |
|
|
|
|
Gcalc_function::op_difference, 2);
|
|
|
|
func.repeat_expression(shape_b);
|
|
|
|
func.repeat_expression(shape_a);
|
|
|
|
break;
|
|
|
|
case SP_TOUCHES_FUNC:
|
2015-06-22 20:33:35 +02:00
|
|
|
if (func.reserve_op_buffer(5))
|
2014-11-27 21:29:37 +01:00
|
|
|
break;
|
2011-09-22 15:53:36 +02:00
|
|
|
func.add_operation(Gcalc_function::op_intersection, 2);
|
|
|
|
func.add_operation(Gcalc_function::v_find_f |
|
|
|
|
Gcalc_function::op_not |
|
|
|
|
Gcalc_function::op_intersection, 2);
|
|
|
|
func.add_operation(Gcalc_function::op_internals, 1);
|
|
|
|
shape_a= func.get_next_expression_pos();
|
2017-10-07 12:17:45 +02:00
|
|
|
if ((null_value= g1.store_shapes(&trn)) ||
|
|
|
|
func.reserve_op_buffer(1))
|
2011-09-22 15:53:36 +02:00
|
|
|
break;
|
|
|
|
func.add_operation(Gcalc_function::op_internals, 1);
|
|
|
|
shape_b= func.get_next_expression_pos();
|
2017-10-07 12:17:45 +02:00
|
|
|
if ((null_value= g2.store_shapes(&trn)) ||
|
|
|
|
func.reserve_op_buffer(1))
|
2011-09-22 15:53:36 +02:00
|
|
|
break;
|
|
|
|
func.add_operation(Gcalc_function::v_find_t |
|
|
|
|
Gcalc_function::op_intersection, 2);
|
|
|
|
func.repeat_expression(shape_a);
|
|
|
|
func.repeat_expression(shape_b);
|
2011-05-04 20:20:17 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
DBUG_ASSERT(FALSE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-09-22 15:53:36 +02:00
|
|
|
if (null_value)
|
2011-05-04 20:20:17 +02:00
|
|
|
goto exit;
|
|
|
|
|
|
|
|
collector.prepare_operation();
|
|
|
|
scan_it.init(&collector);
|
2011-11-20 09:30:43 +01:00
|
|
|
scan_it.killed= (int *) &(current_thd->killed);
|
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
if (func.alloc_states())
|
|
|
|
goto exit;
|
|
|
|
|
2015-06-22 20:33:35 +02:00
|
|
|
result= func.check_function(scan_it);
|
2011-05-04 20:20:17 +02:00
|
|
|
|
|
|
|
exit:
|
|
|
|
collector.reset();
|
|
|
|
func.reset();
|
|
|
|
scan_it.reset();
|
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Item_func_spatial_operation::~Item_func_spatial_operation()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_spatial_operation::val_str(String *str_value)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_func_spatial_operation::val_str");
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
2015-06-26 13:42:49 +02:00
|
|
|
Geometry_ptr_with_buffer_and_mbr g1, g2;
|
2011-05-04 20:20:17 +02:00
|
|
|
uint32 srid= 0;
|
|
|
|
Gcalc_operation_transporter trn(&func, &collector);
|
|
|
|
|
|
|
|
if (func.reserve_op_buffer(1))
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
func.add_operation(spatial_op, 2);
|
|
|
|
|
2015-06-26 13:42:49 +02:00
|
|
|
if ((null_value= (g1.construct(args[0], &tmp_value1) ||
|
|
|
|
g2.construct(args[1], &tmp_value2))))
|
2011-07-04 13:17:34 +02:00
|
|
|
{
|
|
|
|
str_value= 0;
|
2011-05-04 20:20:17 +02:00
|
|
|
goto exit;
|
2011-07-04 13:17:34 +02:00
|
|
|
}
|
2011-05-04 20:20:17 +02:00
|
|
|
|
2015-06-26 13:42:49 +02:00
|
|
|
g1.mbr.add_mbr(&g2.mbr);
|
|
|
|
collector.set_extent(g1.mbr.xmin, g1.mbr.xmax, g1.mbr.ymin, g1.mbr.ymax);
|
2011-05-04 20:20:17 +02:00
|
|
|
|
2015-06-26 13:42:49 +02:00
|
|
|
if ((null_value= g1.store_shapes(&trn) || g2.store_shapes(&trn)))
|
2011-12-08 13:29:45 +01:00
|
|
|
{
|
|
|
|
str_value= 0;
|
|
|
|
goto exit;
|
|
|
|
}
|
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
collector.prepare_operation();
|
|
|
|
if (func.alloc_states())
|
|
|
|
goto exit;
|
|
|
|
|
|
|
|
operation.init(&func);
|
|
|
|
|
|
|
|
if (operation.count_all(&collector) ||
|
|
|
|
operation.get_result(&res_receiver))
|
|
|
|
goto exit;
|
|
|
|
|
|
|
|
|
|
|
|
str_value->set_charset(&my_charset_bin);
|
|
|
|
if (str_value->reserve(SRID_SIZE, 512))
|
|
|
|
goto exit;
|
|
|
|
str_value->length(0);
|
|
|
|
str_value->q_append(srid);
|
|
|
|
|
2015-06-26 13:42:49 +02:00
|
|
|
if (!Geometry::create_from_opresult(&g1.buffer, str_value, res_receiver))
|
2011-05-04 20:20:17 +02:00
|
|
|
goto exit;
|
|
|
|
|
|
|
|
exit:
|
|
|
|
collector.reset();
|
|
|
|
func.reset();
|
|
|
|
res_receiver.reset();
|
|
|
|
DBUG_RETURN(str_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const char *Item_func_spatial_operation::func_name() const
|
|
|
|
{
|
|
|
|
switch (spatial_op) {
|
|
|
|
case Gcalc_function::op_intersection:
|
|
|
|
return "st_intersection";
|
|
|
|
case Gcalc_function::op_difference:
|
|
|
|
return "st_difference";
|
|
|
|
case Gcalc_function::op_union:
|
|
|
|
return "st_union";
|
|
|
|
case Gcalc_function::op_symdifference:
|
|
|
|
return "st_symdifference";
|
|
|
|
default:
|
|
|
|
DBUG_ASSERT(0); // Should never happen
|
|
|
|
return "sp_unknown";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static const int SINUSES_CALCULATED= 32;
|
|
|
|
static double n_sinus[SINUSES_CALCULATED+1]=
|
|
|
|
{
|
|
|
|
0,
|
|
|
|
0.04906767432741802,
|
|
|
|
0.0980171403295606,
|
|
|
|
0.1467304744553618,
|
|
|
|
0.1950903220161283,
|
|
|
|
0.2429801799032639,
|
|
|
|
0.2902846772544623,
|
|
|
|
0.3368898533922201,
|
|
|
|
0.3826834323650898,
|
|
|
|
0.4275550934302821,
|
|
|
|
0.4713967368259976,
|
|
|
|
0.5141027441932217,
|
|
|
|
0.5555702330196022,
|
|
|
|
0.5956993044924334,
|
|
|
|
0.6343932841636455,
|
|
|
|
0.6715589548470183,
|
|
|
|
0.7071067811865475,
|
|
|
|
0.7409511253549591,
|
|
|
|
0.773010453362737,
|
|
|
|
0.8032075314806448,
|
|
|
|
0.8314696123025452,
|
|
|
|
0.8577286100002721,
|
|
|
|
0.8819212643483549,
|
|
|
|
0.9039892931234433,
|
|
|
|
0.9238795325112867,
|
|
|
|
0.9415440651830208,
|
|
|
|
0.9569403357322089,
|
|
|
|
0.970031253194544,
|
|
|
|
0.9807852804032304,
|
|
|
|
0.989176509964781,
|
|
|
|
0.9951847266721968,
|
|
|
|
0.9987954562051724,
|
|
|
|
1
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static void get_n_sincos(int n, double *sinus, double *cosinus)
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(n > 0 && n < SINUSES_CALCULATED*2+1);
|
|
|
|
if (n < (SINUSES_CALCULATED + 1))
|
|
|
|
{
|
|
|
|
*sinus= n_sinus[n];
|
|
|
|
*cosinus= n_sinus[SINUSES_CALCULATED - n];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
n-= SINUSES_CALCULATED;
|
|
|
|
*sinus= n_sinus[SINUSES_CALCULATED - n];
|
|
|
|
*cosinus= -n_sinus[n];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int fill_half_circle(Gcalc_shape_transporter *trn, double x, double y,
|
|
|
|
double ax, double ay)
|
|
|
|
{
|
|
|
|
double n_sin, n_cos;
|
|
|
|
double x_n, y_n;
|
|
|
|
for (int n = 1; n < (SINUSES_CALCULATED * 2 - 1); n++)
|
|
|
|
{
|
|
|
|
get_n_sincos(n, &n_sin, &n_cos);
|
|
|
|
x_n= ax * n_cos - ay * n_sin;
|
|
|
|
y_n= ax * n_sin + ay * n_cos;
|
|
|
|
if (trn->add_point(x_n + x, y_n + y))
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int fill_gap(Gcalc_shape_transporter *trn,
|
|
|
|
double x, double y,
|
|
|
|
double ax, double ay, double bx, double by, double d,
|
|
|
|
bool *empty_gap)
|
|
|
|
{
|
|
|
|
double ab= ax * bx + ay * by;
|
|
|
|
double cosab= ab / (d * d) + GIS_ZERO;
|
|
|
|
double n_sin, n_cos;
|
|
|
|
double x_n, y_n;
|
|
|
|
int n=1;
|
|
|
|
|
|
|
|
*empty_gap= true;
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
get_n_sincos(n++, &n_sin, &n_cos);
|
|
|
|
if (n_cos <= cosab)
|
|
|
|
break;
|
|
|
|
*empty_gap= false;
|
|
|
|
x_n= ax * n_cos - ay * n_sin;
|
|
|
|
y_n= ax * n_sin + ay * n_cos;
|
|
|
|
if (trn->add_point(x_n + x, y_n + y))
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Calculates the vector (p2,p1) and
|
|
|
|
negatively orthogonal to it with the length of d.
|
|
|
|
The result is (ex,ey) - the vector, (px,py) - the orthogonal.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void calculate_perpendicular(
|
|
|
|
double x1, double y1, double x2, double y2, double d,
|
|
|
|
double *ex, double *ey,
|
|
|
|
double *px, double *py)
|
|
|
|
{
|
|
|
|
double q;
|
|
|
|
*ex= x1 - x2;
|
|
|
|
*ey= y1 - y2;
|
|
|
|
q= d / sqrt((*ex) * (*ex) + (*ey) * (*ey));
|
|
|
|
*px= (*ey) * q;
|
|
|
|
*py= -(*ex) * q;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_buffer::Transporter::single_point(double x, double y)
|
|
|
|
{
|
2012-04-29 14:18:38 +02:00
|
|
|
if (buffer_op == Gcalc_function::op_difference)
|
|
|
|
{
|
2017-10-07 12:17:45 +02:00
|
|
|
if (m_fn->reserve_op_buffer(1))
|
|
|
|
return 1;
|
2012-04-29 14:18:38 +02:00
|
|
|
m_fn->add_operation(Gcalc_function::op_false, 0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_nshapes= 0;
|
2011-05-04 20:20:17 +02:00
|
|
|
return add_point_buffer(x, y);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_buffer::Transporter::add_edge_buffer(
|
|
|
|
double x3, double y3, bool round_p1, bool round_p2)
|
|
|
|
{
|
|
|
|
Gcalc_operation_transporter trn(m_fn, m_heap);
|
|
|
|
double e1_x, e1_y, e2_x, e2_y, p1_x, p1_y, p2_x, p2_y;
|
|
|
|
double e1e2;
|
|
|
|
double sin1, cos1;
|
|
|
|
double x_n, y_n;
|
|
|
|
bool empty_gap1, empty_gap2;
|
|
|
|
|
|
|
|
++m_nshapes;
|
|
|
|
if (trn.start_simple_poly())
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
calculate_perpendicular(x1, y1, x2, y2, m_d, &e1_x, &e1_y, &p1_x, &p1_y);
|
|
|
|
calculate_perpendicular(x3, y3, x2, y2, m_d, &e2_x, &e2_y, &p2_x, &p2_y);
|
|
|
|
|
|
|
|
e1e2= e1_x * e2_y - e2_x * e1_y;
|
|
|
|
sin1= n_sinus[1];
|
|
|
|
cos1= n_sinus[31];
|
|
|
|
if (e1e2 < 0)
|
|
|
|
{
|
|
|
|
empty_gap2= false;
|
|
|
|
x_n= x2 + p2_x * cos1 - p2_y * sin1;
|
|
|
|
y_n= y2 + p2_y * cos1 + p2_x * sin1;
|
|
|
|
if (fill_gap(&trn, x2, y2, -p1_x,-p1_y, p2_x,p2_y, m_d, &empty_gap1) ||
|
|
|
|
trn.add_point(x2 + p2_x, y2 + p2_y) ||
|
|
|
|
trn.add_point(x_n, y_n))
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
x_n= x2 - p2_x * cos1 - p2_y * sin1;
|
|
|
|
y_n= y2 - p2_y * cos1 + p2_x * sin1;
|
|
|
|
if (trn.add_point(x_n, y_n) ||
|
|
|
|
trn.add_point(x2 - p2_x, y2 - p2_y) ||
|
|
|
|
fill_gap(&trn, x2, y2, -p2_x, -p2_y, p1_x, p1_y, m_d, &empty_gap2))
|
|
|
|
return 1;
|
|
|
|
empty_gap1= false;
|
|
|
|
}
|
|
|
|
if ((!empty_gap2 && trn.add_point(x2 + p1_x, y2 + p1_y)) ||
|
|
|
|
trn.add_point(x1 + p1_x, y1 + p1_y))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (round_p1 && fill_half_circle(&trn, x1, y1, p1_x, p1_y))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (trn.add_point(x1 - p1_x, y1 - p1_y) ||
|
|
|
|
(!empty_gap1 && trn.add_point(x2 - p1_x, y2 - p1_y)))
|
|
|
|
return 1;
|
|
|
|
return trn.complete_simple_poly();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_buffer::Transporter::add_last_edge_buffer()
|
|
|
|
{
|
|
|
|
Gcalc_operation_transporter trn(m_fn, m_heap);
|
|
|
|
double e1_x, e1_y, p1_x, p1_y;
|
|
|
|
|
|
|
|
++m_nshapes;
|
|
|
|
if (trn.start_simple_poly())
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
calculate_perpendicular(x1, y1, x2, y2, m_d, &e1_x, &e1_y, &p1_x, &p1_y);
|
|
|
|
|
|
|
|
if (trn.add_point(x1 + p1_x, y1 + p1_y) ||
|
|
|
|
trn.add_point(x1 - p1_x, y1 - p1_y) ||
|
|
|
|
trn.add_point(x2 - p1_x, y2 - p1_y) ||
|
|
|
|
fill_half_circle(&trn, x2, y2, -p1_x, -p1_y) ||
|
|
|
|
trn.add_point(x2 + p1_x, y2 + p1_y))
|
|
|
|
return 1;
|
|
|
|
return trn.complete_simple_poly();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_buffer::Transporter::add_point_buffer(double x, double y)
|
|
|
|
{
|
|
|
|
Gcalc_operation_transporter trn(m_fn, m_heap);
|
|
|
|
|
|
|
|
m_nshapes++;
|
|
|
|
if (trn.start_simple_poly())
|
|
|
|
return 1;
|
|
|
|
if (trn.add_point(x - m_d, y) ||
|
|
|
|
fill_half_circle(&trn, x, y, -m_d, 0.0) ||
|
|
|
|
trn.add_point(x + m_d, y) ||
|
|
|
|
fill_half_circle(&trn, x, y, m_d, 0.0))
|
|
|
|
return 1;
|
|
|
|
return trn.complete_simple_poly();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_buffer::Transporter::start_line()
|
|
|
|
{
|
2011-07-04 13:03:36 +02:00
|
|
|
if (buffer_op == Gcalc_function::op_difference)
|
|
|
|
{
|
2014-02-18 14:15:25 +01:00
|
|
|
if (m_fn->reserve_op_buffer(1))
|
|
|
|
return 1;
|
2012-04-29 14:18:38 +02:00
|
|
|
m_fn->add_operation(Gcalc_function::op_false, 0);
|
2011-07-04 13:03:36 +02:00
|
|
|
skip_line= TRUE;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_nshapes= 0;
|
|
|
|
|
|
|
|
if (m_fn->reserve_op_buffer(2))
|
|
|
|
return 1;
|
2011-09-22 15:53:36 +02:00
|
|
|
last_shape_pos= m_fn->get_next_expression_pos();
|
2011-07-04 13:03:36 +02:00
|
|
|
m_fn->add_operation(buffer_op, 0);
|
2011-05-04 20:20:17 +02:00
|
|
|
m_npoints= 0;
|
|
|
|
int_start_line();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_buffer::Transporter::start_poly()
|
|
|
|
{
|
2011-07-04 13:03:36 +02:00
|
|
|
m_nshapes= 1;
|
|
|
|
|
|
|
|
if (m_fn->reserve_op_buffer(2))
|
|
|
|
return 1;
|
2011-09-22 15:53:36 +02:00
|
|
|
last_shape_pos= m_fn->get_next_expression_pos();
|
2011-07-04 13:03:36 +02:00
|
|
|
m_fn->add_operation(buffer_op, 0);
|
2011-05-04 20:20:17 +02:00
|
|
|
return Gcalc_operation_transporter::start_poly();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-04 13:03:36 +02:00
|
|
|
int Item_func_buffer::Transporter::complete_poly()
|
|
|
|
{
|
|
|
|
if (Gcalc_operation_transporter::complete_poly())
|
|
|
|
return 1;
|
|
|
|
m_fn->add_operands_to_op(last_shape_pos, m_nshapes);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
int Item_func_buffer::Transporter::start_ring()
|
|
|
|
{
|
|
|
|
m_npoints= 0;
|
|
|
|
return Gcalc_operation_transporter::start_ring();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-04 13:03:36 +02:00
|
|
|
int Item_func_buffer::Transporter::start_collection(int n_objects)
|
|
|
|
{
|
|
|
|
if (m_fn->reserve_op_buffer(1))
|
|
|
|
return 1;
|
|
|
|
m_fn->add_operation(Gcalc_function::op_union, n_objects);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
int Item_func_buffer::Transporter::add_point(double x, double y)
|
|
|
|
{
|
2011-07-04 13:03:36 +02:00
|
|
|
if (skip_line)
|
|
|
|
return 0;
|
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
if (m_npoints && x == x2 && y == y2)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
++m_npoints;
|
|
|
|
|
|
|
|
if (m_npoints == 1)
|
|
|
|
{
|
|
|
|
x00= x;
|
|
|
|
y00= y;
|
|
|
|
}
|
|
|
|
else if (m_npoints == 2)
|
|
|
|
{
|
|
|
|
x01= x;
|
|
|
|
y01= y;
|
|
|
|
}
|
|
|
|
else if (add_edge_buffer(x, y, (m_npoints == 3) && line_started(), false))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
x1= x2;
|
|
|
|
y1= y2;
|
|
|
|
x2= x;
|
|
|
|
y2= y;
|
|
|
|
|
|
|
|
return line_started() ? 0 : Gcalc_operation_transporter::add_point(x, y);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_buffer::Transporter::complete()
|
|
|
|
{
|
|
|
|
if (m_npoints)
|
|
|
|
{
|
|
|
|
if (m_npoints == 1)
|
|
|
|
{
|
|
|
|
if (add_point_buffer(x2, y2))
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else if (m_npoints == 2)
|
|
|
|
{
|
|
|
|
if (add_edge_buffer(x1, y1, true, true))
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else if (line_started())
|
|
|
|
{
|
|
|
|
if (add_last_edge_buffer())
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-07-07 18:30:51 +02:00
|
|
|
if (x2 != x00 || y2 != y00)
|
2011-05-04 20:20:17 +02:00
|
|
|
{
|
|
|
|
if (add_edge_buffer(x00, y00, false, false))
|
|
|
|
return 1;
|
|
|
|
x1= x2;
|
|
|
|
y1= y2;
|
|
|
|
x2= x00;
|
|
|
|
y2= y00;
|
|
|
|
}
|
|
|
|
if (add_edge_buffer(x01, y01, false, false))
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_buffer::Transporter::complete_line()
|
|
|
|
{
|
2011-07-04 13:03:36 +02:00
|
|
|
if (!skip_line)
|
|
|
|
{
|
|
|
|
if (complete())
|
|
|
|
return 1;
|
|
|
|
int_complete_line();
|
|
|
|
m_fn->add_operands_to_op(last_shape_pos, m_nshapes);
|
|
|
|
}
|
|
|
|
skip_line= FALSE;
|
2011-05-04 20:20:17 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_buffer::Transporter::complete_ring()
|
|
|
|
{
|
|
|
|
return complete() ||
|
|
|
|
Gcalc_operation_transporter::complete_ring();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_buffer::val_str(String *str_value)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_func_buffer::val_str");
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
MDEV-10306 Wrong results with combination of CONCAT, SUBSTR and CONVERT in subquery
The bug happens because of a combination of unfortunate circumstances:
1. Arguments args[0] and args[2] of Item_func_concat point recursively
(through Item_direct_view_ref's) to the same Item_func_conv_charset.
Both args[0]->args[0]->ref[0] and args[2]->args[0]->ref[0] refer to
this Item_func_conv_charset.
2. When Item_func_concat::args[0]->val_str() is called,
Item_func_conv_charset::val_str() writes its result to
Item_func_conc_charset::tmp_value.
3. Then, for optimization purposes (to avoid copying),
Item_func_substr::val_str() initializes Item_func_substr::tmp_value
to point to the buffer fragment owned by Item_func_conv_charset::tmp_value
Item_func_substr::tmp_value is returned as a result of
Item_func_concat::args[0]->val_str().
4. Due to optimization to avoid memory reallocs,
Item_func_concat::val_str() remembers the result of args[0]->val_str()
in "res" and further uses "res" to collect the return value.
5. When Item_func_concat::args[2]->val_str() is called,
Item_func_conv_charset::tmp_value gets overwritten (see #1),
which effectively overwrites args[0]'s Item_func_substr::tmp_value (see #3),
which effectively overwrites "res" (see #4).
This patch does the following:
a. Changes Item_func_conv_charset::val_str(String *str) to use
tmp_value and str the other way around. After this change tmp_value
is used to store a temporary result, while str is used to return the value.
The fixes the second problem (without SUBSTR):
SELECT CONCAT(t2,'-',t2) c2
FROM (SELECT CONVERT(t USING latin1) t2 FROM t1) sub;
As Item_func_concat::val_str() supplies two different buffers when calling
args[0]->val_str() and args[2]->val_str(), in the new reduction the result
created during args[0]->val_str() does not get overwritten by
args[2]->val_str().
b. Fixing the same problem in val_str() for similar classes
Item_func_to_base64
Item_func_from_base64
Item_func_weight_string
Item_func_hex
Item_func_unhex
Item_func_quote
Item_func_compress
Item_func_uncompress
Item_func_des_encrypt
Item_func_des_decrypt
Item_func_conv_charset
Item_func_reverse
Item_func_soundex
Item_func_aes_encrypt
Item_func_aes_decrypt
Item_func_buffer
c. Fixing Item_func::val_str_from_val_str_ascii() the same way.
Now Item_str_ascii_func::ascii_buff is used for temporary value,
while the parameter passed to val_str() is used to return the result.
This fixes the same problem when conversion (from ASCII to e.g. UCS2)
takes place. See the ctype_ucs.test for example queries that returned
wrong results before the fix.
d. Some Item_func descendand classes had temporary String buffers
(tmp_value and tmp_str), but did not really use them.
Removing these temporary buffers from:
Item_func_decode_histogram
Item_func_format
Item_func_binlog_gtid_pos
Item_func_spatial_collection:
e. Removing Item_func_buffer::tmp_value, because it's not used any more.
f. Renaming Item_func_[un]compress::buffer to "tmp_value",
for consistency with other classes.
Note, this patch does not fix the following classes
(although they have a similar problem):
Item_str_conv
Item_func_make_set
Item_char_typecast
They have a complex implementations and simple swapping between "tmp_value"
and "str" won't work. These classes will be fixed separately.
2017-06-19 10:45:32 +02:00
|
|
|
String *obj= args[0]->val_str(str_value);
|
2011-05-04 20:20:17 +02:00
|
|
|
double dist= args[1]->val_real();
|
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *g;
|
|
|
|
uint32 srid= 0;
|
|
|
|
String *str_result= NULL;
|
|
|
|
Transporter trn(&func, &collector, dist);
|
2011-12-08 13:29:45 +01:00
|
|
|
MBR mbr;
|
|
|
|
const char *c_end;
|
2011-05-04 20:20:17 +02:00
|
|
|
|
|
|
|
null_value= 1;
|
|
|
|
if (args[0]->null_value || args[1]->null_value ||
|
2011-12-08 13:29:45 +01:00
|
|
|
!(g= Geometry::construct(&buffer, obj->ptr(), obj->length())) ||
|
|
|
|
g->get_mbr(&mbr, &c_end))
|
2011-05-04 20:20:17 +02:00
|
|
|
goto mem_error;
|
|
|
|
|
2011-12-08 13:29:45 +01:00
|
|
|
if (dist > 0.0)
|
|
|
|
mbr.buffer(dist);
|
2015-02-10 13:16:31 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* This happens when dist is too far negative. */
|
|
|
|
if (mbr.xmax + dist < mbr.xmin || mbr.ymax + dist < mbr.ymin)
|
|
|
|
goto return_empty_result;
|
|
|
|
}
|
|
|
|
|
2011-12-08 13:29:45 +01:00
|
|
|
collector.set_extent(mbr.xmin, mbr.xmax, mbr.ymin, mbr.ymax);
|
2011-06-30 16:24:52 +02:00
|
|
|
/*
|
|
|
|
If the distance given is 0, the Buffer function is in fact NOOP,
|
|
|
|
so it's natural just to return the argument1.
|
|
|
|
Besides, internal calculations here can't handle zero distance anyway.
|
|
|
|
*/
|
|
|
|
if (fabs(dist) < GIS_ZERO)
|
|
|
|
{
|
|
|
|
null_value= 0;
|
|
|
|
str_result= obj;
|
|
|
|
goto mem_error;
|
|
|
|
}
|
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
if (g->store_shapes(&trn))
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
collector.prepare_operation();
|
|
|
|
if (func.alloc_states())
|
|
|
|
goto mem_error;
|
|
|
|
operation.init(&func);
|
2011-11-20 09:30:43 +01:00
|
|
|
operation.killed= (int *) &(current_thd->killed);
|
2011-05-04 20:20:17 +02:00
|
|
|
|
|
|
|
if (operation.count_all(&collector) ||
|
|
|
|
operation.get_result(&res_receiver))
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
|
2015-02-10 13:16:31 +01:00
|
|
|
return_empty_result:
|
2011-05-04 20:20:17 +02:00
|
|
|
str_value->set_charset(&my_charset_bin);
|
|
|
|
if (str_value->reserve(SRID_SIZE, 512))
|
|
|
|
goto mem_error;
|
|
|
|
str_value->length(0);
|
|
|
|
str_value->q_append(srid);
|
|
|
|
|
2014-11-27 21:29:37 +01:00
|
|
|
if (!Geometry::create_from_opresult(&buffer, str_value, res_receiver))
|
2011-05-04 20:20:17 +02:00
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
null_value= 0;
|
|
|
|
str_result= str_value;
|
|
|
|
mem_error:
|
|
|
|
collector.reset();
|
|
|
|
func.reset();
|
|
|
|
res_receiver.reset();
|
|
|
|
DBUG_RETURN(str_result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
longlong Item_func_isempty::val_int()
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2007-02-21 11:45:19 +01:00
|
|
|
String tmp;
|
|
|
|
String *swkb= args[0]->val_str(&tmp);
|
|
|
|
Geometry_buffer buffer;
|
|
|
|
|
|
|
|
null_value= args[0]->null_value ||
|
|
|
|
!(Geometry::construct(&buffer, swkb->ptr(), swkb->length()));
|
|
|
|
return null_value ? 1 : 0;
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
longlong Item_func_issimple::val_int()
|
|
|
|
{
|
2007-02-21 11:45:19 +01:00
|
|
|
String *swkb= args[0]->val_str(&tmp);
|
|
|
|
Geometry_buffer buffer;
|
2011-05-04 20:20:17 +02:00
|
|
|
Gcalc_operation_transporter trn(&func, &collector);
|
|
|
|
Geometry *g;
|
|
|
|
int result= 1;
|
2011-12-08 13:29:45 +01:00
|
|
|
MBR mbr;
|
|
|
|
const char *c_end;
|
2011-05-04 20:20:17 +02:00
|
|
|
|
|
|
|
DBUG_ENTER("Item_func_issimple::val_int");
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
2007-02-21 11:45:19 +01:00
|
|
|
|
2015-06-23 08:57:05 +02:00
|
|
|
null_value= 0;
|
|
|
|
if ((args[0]->null_value ||
|
|
|
|
!(g= Geometry::construct(&buffer, swkb->ptr(), swkb->length())) ||
|
|
|
|
g->get_mbr(&mbr, &c_end)))
|
|
|
|
{
|
|
|
|
/* We got NULL as an argument. Have to return -1 */
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
2011-05-04 20:20:17 +02:00
|
|
|
|
2011-12-08 13:29:45 +01:00
|
|
|
collector.set_extent(mbr.xmin, mbr.xmax, mbr.ymin, mbr.ymax);
|
2011-05-04 20:20:17 +02:00
|
|
|
|
|
|
|
if (g->get_class_info()->m_type_id == Geometry::wkb_point)
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
|
|
|
|
if (g->store_shapes(&trn))
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
collector.prepare_operation();
|
|
|
|
scan_it.init(&collector);
|
|
|
|
|
|
|
|
while (scan_it.more_points())
|
|
|
|
{
|
2015-03-13 13:10:31 +01:00
|
|
|
const Gcalc_scan_iterator::event_point *ev, *next_ev;
|
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
if (scan_it.step())
|
|
|
|
goto mem_error;
|
|
|
|
|
2011-09-01 08:44:56 +02:00
|
|
|
ev= scan_it.get_events();
|
|
|
|
if (ev->simple_event())
|
|
|
|
continue;
|
|
|
|
|
2015-03-13 13:10:31 +01:00
|
|
|
next_ev= ev->get_next();
|
|
|
|
if ((ev->event & (scev_thread | scev_single_point)) && !next_ev)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if ((ev->event == scev_two_threads) && !next_ev->get_next())
|
2011-09-01 08:44:56 +02:00
|
|
|
continue;
|
|
|
|
|
2015-03-13 13:10:31 +01:00
|
|
|
/* If the first and last points of a curve coincide - that is */
|
|
|
|
/* an exception to the rule and the line is considered as simple. */
|
|
|
|
if ((next_ev && !next_ev->get_next()) &&
|
|
|
|
(ev->event & (scev_thread | scev_end)) &&
|
|
|
|
(next_ev->event & (scev_thread | scev_end)))
|
2011-09-01 08:44:56 +02:00
|
|
|
continue;
|
|
|
|
|
|
|
|
result= 0;
|
|
|
|
break;
|
2011-05-04 20:20:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
collector.reset();
|
|
|
|
func.reset();
|
|
|
|
scan_it.reset();
|
|
|
|
DBUG_RETURN(result);
|
|
|
|
mem_error:
|
|
|
|
null_value= 1;
|
|
|
|
DBUG_RETURN(0);
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
longlong Item_func_isclosed::val_int()
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-30 12:22:34 +02:00
|
|
|
String tmp;
|
|
|
|
String *swkb= args[0]->val_str(&tmp);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom;
|
2004-03-04 07:50:37 +01:00
|
|
|
int isclosed= 0; // In case of error
|
2003-05-30 12:22:34 +02:00
|
|
|
|
2015-06-23 08:57:05 +02:00
|
|
|
null_value= 0;
|
|
|
|
if (!swkb ||
|
|
|
|
args[0]->null_value ||
|
|
|
|
!(geom= Geometry::construct(&buffer, swkb->ptr(), swkb->length())) ||
|
|
|
|
geom->is_closed(&isclosed))
|
|
|
|
{
|
|
|
|
/* IsClosed(NULL) should return -1 */
|
|
|
|
return -1;
|
|
|
|
}
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
return (longlong) isclosed;
|
|
|
|
}
|
|
|
|
|
2014-11-27 21:29:37 +01:00
|
|
|
|
|
|
|
longlong Item_func_isring::val_int()
|
|
|
|
{
|
|
|
|
/* It's actually a combination of two functions - IsClosed and IsSimple */
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
|
|
|
String tmp;
|
|
|
|
String *swkb= args[0]->val_str(&tmp);
|
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom;
|
|
|
|
int isclosed= 0; // In case of error
|
|
|
|
|
2015-06-23 08:57:05 +02:00
|
|
|
null_value= 0;
|
|
|
|
if (!swkb ||
|
|
|
|
args[0]->null_value ||
|
|
|
|
!(geom= Geometry::construct(&buffer, swkb->ptr(), swkb->length())) ||
|
|
|
|
geom->is_closed(&isclosed))
|
|
|
|
{
|
|
|
|
/* IsRing(NULL) should return -1 */
|
|
|
|
return -1;
|
|
|
|
}
|
2014-11-27 21:29:37 +01:00
|
|
|
|
|
|
|
if (!isclosed)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return Item_func_issimple::val_int();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
/*
|
|
|
|
Numerical functions
|
|
|
|
*/
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
longlong Item_func_dimension::val_int()
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2004-03-04 07:50:37 +01:00
|
|
|
uint32 dim= 0; // In case of error
|
2003-05-30 12:22:34 +02:00
|
|
|
String *swkb= args[0]->val_str(&value);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom;
|
2004-03-04 07:50:37 +01:00
|
|
|
const char *dummy;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
null_value= (!swkb ||
|
|
|
|
args[0]->null_value ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer, swkb->ptr(), swkb->length())) ||
|
2004-03-12 09:04:00 +01:00
|
|
|
geom->dimension(&dim, &dummy));
|
2003-05-30 12:22:34 +02:00
|
|
|
return (longlong) dim;
|
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
longlong Item_func_numinteriorring::val_int()
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2004-03-04 07:50:37 +01:00
|
|
|
uint32 num= 0; // In case of error
|
2003-05-30 12:22:34 +02:00
|
|
|
String *swkb= args[0]->val_str(&value);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
null_value= (!swkb ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer,
|
|
|
|
swkb->ptr(), swkb->length())) ||
|
2004-03-12 09:04:00 +01:00
|
|
|
geom->num_interior_ring(&num));
|
2003-05-30 12:22:34 +02:00
|
|
|
return (longlong) num;
|
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
longlong Item_func_numgeometries::val_int()
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2004-03-04 07:50:37 +01:00
|
|
|
uint32 num= 0; // In case of errors
|
2003-05-30 12:22:34 +02:00
|
|
|
String *swkb= args[0]->val_str(&value);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
null_value= (!swkb ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer,
|
|
|
|
swkb->ptr(), swkb->length())) ||
|
2004-03-12 09:04:00 +01:00
|
|
|
geom->num_geometries(&num));
|
2003-05-30 12:22:34 +02:00
|
|
|
return (longlong) num;
|
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2003-05-30 12:22:34 +02:00
|
|
|
longlong Item_func_numpoints::val_int()
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2004-03-04 07:50:37 +01:00
|
|
|
uint32 num= 0; // In case of errors
|
2003-05-30 12:22:34 +02:00
|
|
|
String *swkb= args[0]->val_str(&value);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
null_value= (!swkb ||
|
|
|
|
args[0]->null_value ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer,
|
|
|
|
swkb->ptr(), swkb->length())) ||
|
2004-03-12 09:04:00 +01:00
|
|
|
geom->num_points(&num));
|
2003-05-30 12:22:34 +02:00
|
|
|
return (longlong) num;
|
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2004-11-11 19:39:35 +01:00
|
|
|
double Item_func_x::val_real()
|
2003-05-30 12:22:34 +02:00
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2004-03-04 07:50:37 +01:00
|
|
|
double res= 0.0; // In case of errors
|
2003-05-30 12:22:34 +02:00
|
|
|
String *swkb= args[0]->val_str(&value);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
null_value= (!swkb ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer,
|
|
|
|
swkb->ptr(), swkb->length())) ||
|
2004-03-12 09:04:00 +01:00
|
|
|
geom->get_x(&res));
|
2003-05-30 12:22:34 +02:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2004-11-11 19:39:35 +01:00
|
|
|
double Item_func_y::val_real()
|
2003-05-30 12:22:34 +02:00
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2004-03-04 07:50:37 +01:00
|
|
|
double res= 0; // In case of errors
|
2003-05-30 12:22:34 +02:00
|
|
|
String *swkb= args[0]->val_str(&value);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
null_value= (!swkb ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer,
|
|
|
|
swkb->ptr(), swkb->length())) ||
|
2004-03-12 09:04:00 +01:00
|
|
|
geom->get_y(&res));
|
2003-05-30 12:22:34 +02:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2004-03-04 07:50:37 +01:00
|
|
|
|
2004-11-11 19:39:35 +01:00
|
|
|
double Item_func_area::val_real()
|
2003-05-30 12:22:34 +02:00
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2004-03-04 07:50:37 +01:00
|
|
|
double res= 0; // In case of errors
|
2003-05-30 12:22:34 +02:00
|
|
|
String *swkb= args[0]->val_str(&value);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom;
|
2004-03-04 07:50:37 +01:00
|
|
|
const char *dummy;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
null_value= (!swkb ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer,
|
|
|
|
swkb->ptr(), swkb->length())) ||
|
2004-03-12 09:04:00 +01:00
|
|
|
geom->area(&res, &dummy));
|
2003-05-30 12:22:34 +02:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2004-11-11 19:39:35 +01:00
|
|
|
double Item_func_glength::val_real()
|
2003-05-30 12:22:34 +02:00
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2004-03-04 07:50:37 +01:00
|
|
|
double res= 0; // In case of errors
|
2003-05-30 12:22:34 +02:00
|
|
|
String *swkb= args[0]->val_str(&value);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *geom;
|
2011-09-21 10:26:21 +02:00
|
|
|
const char *end;
|
2003-05-30 12:22:34 +02:00
|
|
|
|
|
|
|
null_value= (!swkb ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!(geom= Geometry::construct(&buffer,
|
2006-12-14 23:51:37 +01:00
|
|
|
swkb->ptr(),
|
|
|
|
swkb->length())) ||
|
2011-09-21 10:26:21 +02:00
|
|
|
geom->geom_length(&res, &end));
|
2003-05-30 12:22:34 +02:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
longlong Item_func_srid::val_int()
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-30 12:22:34 +02:00
|
|
|
String *swkb= args[0]->val_str(&value);
|
2004-03-12 09:04:00 +01:00
|
|
|
Geometry_buffer buffer;
|
2004-05-05 07:59:17 +02:00
|
|
|
|
2004-04-05 12:56:05 +02:00
|
|
|
null_value= (!swkb ||
|
2005-09-21 11:35:01 +02:00
|
|
|
!Geometry::construct(&buffer,
|
|
|
|
swkb->ptr(), swkb->length()));
|
2004-03-04 07:50:37 +01:00
|
|
|
if (null_value)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return (longlong) (uint4korr(swkb->ptr()));
|
2003-05-30 12:22:34 +02:00
|
|
|
}
|
2004-01-15 18:06:22 +01:00
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
|
|
|
|
double Item_func_distance::val_real()
|
|
|
|
{
|
2011-11-18 15:15:06 +01:00
|
|
|
bool cur_point_edge;
|
2011-05-04 20:20:17 +02:00
|
|
|
const Gcalc_scan_iterator::point *evpos;
|
|
|
|
const Gcalc_heap::Info *cur_point, *dist_point;
|
2011-10-14 13:10:55 +02:00
|
|
|
const Gcalc_scan_iterator::event_point *ev;
|
2011-05-04 20:20:17 +02:00
|
|
|
double t, distance, cur_distance;
|
|
|
|
double x1, x2, y1, y2;
|
|
|
|
double ex, ey, vx, vy, e_sqrlen;
|
|
|
|
uint obj2_si;
|
|
|
|
Gcalc_operation_transporter trn(&func, &collector);
|
|
|
|
|
|
|
|
DBUG_ENTER("Item_func_distance::val_real");
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
|
|
|
String *res1= args[0]->val_str(&tmp_value1);
|
|
|
|
String *res2= args[1]->val_str(&tmp_value2);
|
|
|
|
Geometry_buffer buffer1, buffer2;
|
|
|
|
Geometry *g1, *g2;
|
2011-12-08 13:29:45 +01:00
|
|
|
MBR mbr1, mbr2;
|
|
|
|
const char *c_end;
|
2011-05-04 20:20:17 +02:00
|
|
|
|
|
|
|
|
|
|
|
if ((null_value= (args[0]->null_value || args[1]->null_value ||
|
|
|
|
!(g1= Geometry::construct(&buffer1, res1->ptr(), res1->length())) ||
|
2013-03-22 14:32:27 +01:00
|
|
|
!(g2= Geometry::construct(&buffer2, res2->ptr(), res2->length())) ||
|
|
|
|
g1->get_mbr(&mbr1, &c_end) ||
|
|
|
|
g2->get_mbr(&mbr2, &c_end))))
|
2011-05-04 20:20:17 +02:00
|
|
|
goto mem_error;
|
|
|
|
|
2011-12-08 13:29:45 +01:00
|
|
|
mbr1.add_mbr(&mbr2);
|
|
|
|
collector.set_extent(mbr1.xmin, mbr1.xmax, mbr1.ymin, mbr1.ymax);
|
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
if ((g1->get_class_info()->m_type_id == Geometry::wkb_point) &&
|
|
|
|
(g2->get_class_info()->m_type_id == Geometry::wkb_point))
|
|
|
|
{
|
|
|
|
if (((Gis_point *) g1)->get_xy(&x1, &y1) ||
|
|
|
|
((Gis_point *) g2)->get_xy(&x2, &y2))
|
|
|
|
goto mem_error;
|
|
|
|
ex= x2 - x1;
|
|
|
|
ey= y2 - y1;
|
|
|
|
DBUG_RETURN(sqrt(ex * ex + ey * ey));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (func.reserve_op_buffer(1))
|
|
|
|
goto mem_error;
|
|
|
|
func.add_operation(Gcalc_function::op_intersection, 2);
|
|
|
|
|
|
|
|
if (g1->store_shapes(&trn))
|
|
|
|
goto mem_error;
|
|
|
|
obj2_si= func.get_nshapes();
|
|
|
|
if (g2->store_shapes(&trn) || func.alloc_states())
|
|
|
|
goto mem_error;
|
|
|
|
|
2011-09-13 10:59:11 +02:00
|
|
|
if (obj2_si == 0 || func.get_nshapes() == obj2_si)
|
|
|
|
{
|
|
|
|
distance= 0.0;
|
|
|
|
null_value= 1;
|
|
|
|
goto exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-04 20:20:17 +02:00
|
|
|
collector.prepare_operation();
|
|
|
|
scan_it.init(&collector);
|
|
|
|
|
|
|
|
distance= DBL_MAX;
|
|
|
|
while (scan_it.more_points())
|
|
|
|
{
|
|
|
|
if (scan_it.step())
|
|
|
|
goto mem_error;
|
|
|
|
evpos= scan_it.get_event_position();
|
2011-09-01 08:44:56 +02:00
|
|
|
ev= scan_it.get_events();
|
2011-05-04 20:20:17 +02:00
|
|
|
|
2011-09-01 08:44:56 +02:00
|
|
|
if (ev->simple_event())
|
2011-05-04 20:20:17 +02:00
|
|
|
{
|
2011-09-01 08:44:56 +02:00
|
|
|
cur_point= ev->pi;
|
|
|
|
goto count_distance;
|
2011-05-04 20:20:17 +02:00
|
|
|
}
|
|
|
|
/*
|
2011-09-01 08:44:56 +02:00
|
|
|
handling intersection we only need to check if it's the intersecion
|
|
|
|
of objects 1 and 2. In this case distance is 0
|
2011-05-04 20:20:17 +02:00
|
|
|
*/
|
2011-09-01 08:44:56 +02:00
|
|
|
cur_point= NULL;
|
2011-05-04 20:20:17 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
having these events we need to check for possible intersection
|
|
|
|
of objects
|
|
|
|
scev_thread | scev_two_threads | scev_single_point
|
|
|
|
*/
|
2011-09-22 15:53:36 +02:00
|
|
|
func.clear_i_states();
|
2011-05-04 20:20:17 +02:00
|
|
|
for (Gcalc_point_iterator pit(&scan_it); pit.point() != evpos; ++pit)
|
|
|
|
{
|
|
|
|
gcalc_shape_info si= pit.point()->get_shape();
|
|
|
|
if ((func.get_shape_kind(si) == Gcalc_function::shape_polygon))
|
2011-09-22 15:53:36 +02:00
|
|
|
func.invert_i_state(si);
|
2011-05-04 20:20:17 +02:00
|
|
|
}
|
2011-09-01 08:44:56 +02:00
|
|
|
|
2011-09-22 15:53:36 +02:00
|
|
|
func.clear_b_states();
|
2011-09-01 08:44:56 +02:00
|
|
|
for (; ev; ev= ev->get_next())
|
2011-05-04 20:20:17 +02:00
|
|
|
{
|
2011-09-01 08:44:56 +02:00
|
|
|
if (ev->event != scev_intersection)
|
|
|
|
cur_point= ev->pi;
|
2011-09-22 15:53:36 +02:00
|
|
|
func.set_b_state(ev->get_shape());
|
2011-09-01 08:44:56 +02:00
|
|
|
if (func.count())
|
|
|
|
{
|
|
|
|
/* Point of one object is inside the other - intersection found */
|
|
|
|
distance= 0;
|
|
|
|
goto exit;
|
|
|
|
}
|
2011-05-04 20:20:17 +02:00
|
|
|
}
|
|
|
|
|
2011-09-01 08:44:56 +02:00
|
|
|
if (!cur_point)
|
|
|
|
continue;
|
2011-05-04 20:20:17 +02:00
|
|
|
|
|
|
|
count_distance:
|
2016-02-14 20:57:48 +01:00
|
|
|
if (cur_point->node.shape.shape >= obj2_si)
|
2011-05-04 20:20:17 +02:00
|
|
|
continue;
|
|
|
|
cur_point_edge= !cur_point->is_bottom();
|
|
|
|
|
|
|
|
for (dist_point= collector.get_first(); dist_point; dist_point= dist_point->get_next())
|
|
|
|
{
|
|
|
|
/* We only check vertices of object 2 */
|
2012-08-31 16:50:45 +02:00
|
|
|
if (dist_point->type != Gcalc_heap::nt_shape_node ||
|
2016-02-14 20:57:48 +01:00
|
|
|
dist_point->node.shape.shape < obj2_si)
|
2011-05-04 20:20:17 +02:00
|
|
|
continue;
|
|
|
|
|
|
|
|
/* if we have an edge to check */
|
2016-02-14 20:57:48 +01:00
|
|
|
if (dist_point->node.shape.left)
|
2011-05-04 20:20:17 +02:00
|
|
|
{
|
2016-02-14 20:57:48 +01:00
|
|
|
t= count_edge_t(dist_point, dist_point->node.shape.left, cur_point,
|
2011-05-04 20:20:17 +02:00
|
|
|
ex, ey, vx, vy, e_sqrlen);
|
|
|
|
if ((t>0.0) && (t<1.0))
|
|
|
|
{
|
|
|
|
cur_distance= distance_to_line(ex, ey, vx, vy, e_sqrlen);
|
|
|
|
if (distance > cur_distance)
|
|
|
|
distance= cur_distance;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (cur_point_edge)
|
|
|
|
{
|
2016-02-14 20:57:48 +01:00
|
|
|
t= count_edge_t(cur_point, cur_point->node.shape.left, dist_point,
|
2011-05-04 20:20:17 +02:00
|
|
|
ex, ey, vx, vy, e_sqrlen);
|
|
|
|
if ((t>0.0) && (t<1.0))
|
|
|
|
{
|
|
|
|
cur_distance= distance_to_line(ex, ey, vx, vy, e_sqrlen);
|
|
|
|
if (distance > cur_distance)
|
|
|
|
distance= cur_distance;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
cur_distance= distance_points(cur_point, dist_point);
|
|
|
|
if (distance > cur_distance)
|
|
|
|
distance= cur_distance;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
exit:
|
|
|
|
collector.reset();
|
|
|
|
func.reset();
|
|
|
|
scan_it.reset();
|
|
|
|
DBUG_RETURN(distance);
|
|
|
|
mem_error:
|
|
|
|
null_value= 1;
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-11-27 21:29:37 +01:00
|
|
|
String *Item_func_pointonsurface::val_str(String *str)
|
|
|
|
{
|
|
|
|
Gcalc_operation_transporter trn(&func, &collector);
|
|
|
|
|
|
|
|
DBUG_ENTER("Item_func_pointonsurface::val_real");
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
|
|
|
String *res= args[0]->val_str(&tmp_value);
|
|
|
|
Geometry_buffer buffer;
|
|
|
|
Geometry *g;
|
|
|
|
MBR mbr;
|
|
|
|
const char *c_end;
|
2015-08-25 09:46:31 +02:00
|
|
|
double UNINIT_VAR(px), UNINIT_VAR(py), x0, y0;
|
2014-11-27 21:29:37 +01:00
|
|
|
String *result= 0;
|
|
|
|
const Gcalc_scan_iterator::point *pprev= NULL;
|
|
|
|
uint32 srid;
|
|
|
|
|
|
|
|
null_value= 1;
|
|
|
|
if ((args[0]->null_value ||
|
|
|
|
!(g= Geometry::construct(&buffer, res->ptr(), res->length())) ||
|
|
|
|
g->get_mbr(&mbr, &c_end)))
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
collector.set_extent(mbr.xmin, mbr.xmax, mbr.ymin, mbr.ymax);
|
|
|
|
|
|
|
|
if (g->store_shapes(&trn))
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
collector.prepare_operation();
|
|
|
|
scan_it.init(&collector);
|
|
|
|
|
|
|
|
while (scan_it.more_points())
|
|
|
|
{
|
|
|
|
if (scan_it.step())
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
if (scan_it.get_h() > GIS_ZERO)
|
|
|
|
{
|
|
|
|
y0= scan_it.get_y();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!scan_it.more_points())
|
|
|
|
{
|
|
|
|
goto exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (scan_it.step())
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
for (Gcalc_point_iterator pit(&scan_it); pit.point(); ++pit)
|
|
|
|
{
|
|
|
|
if (pprev == NULL)
|
|
|
|
{
|
|
|
|
pprev= pit.point();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
x0= scan_it.get_sp_x(pprev);
|
|
|
|
px= scan_it.get_sp_x(pit.point());
|
2015-03-15 19:20:38 +01:00
|
|
|
if (px - x0 > GIS_ZERO)
|
2014-11-27 21:29:37 +01:00
|
|
|
{
|
|
|
|
if (scan_it.get_h() > GIS_ZERO)
|
|
|
|
{
|
|
|
|
px= (px + x0) / 2.0;
|
|
|
|
py= scan_it.get_y();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
px= (px + x0) / 2.0;
|
|
|
|
py= (y0 + scan_it.get_y()) / 2.0;
|
|
|
|
}
|
|
|
|
null_value= 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
pprev= NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (null_value)
|
|
|
|
goto exit;
|
|
|
|
|
|
|
|
str->set_charset(&my_charset_bin);
|
|
|
|
if (str->reserve(SRID_SIZE, 512))
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
str->length(0);
|
|
|
|
srid= uint4korr(res->ptr());
|
|
|
|
str->q_append(srid);
|
|
|
|
|
|
|
|
if (Geometry::create_point(str, px, py))
|
|
|
|
goto mem_error;
|
|
|
|
|
|
|
|
result= str;
|
|
|
|
|
|
|
|
exit:
|
|
|
|
collector.reset();
|
|
|
|
func.reset();
|
|
|
|
scan_it.reset();
|
|
|
|
DBUG_RETURN(result);
|
|
|
|
|
|
|
|
mem_error:
|
|
|
|
collector.reset();
|
|
|
|
func.reset();
|
|
|
|
scan_it.reset();
|
|
|
|
null_value= 1;
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-30 21:30:24 +02:00
|
|
|
#ifndef DBUG_OFF
|
|
|
|
longlong Item_func_gis_debug::val_int()
|
|
|
|
{
|
|
|
|
/* For now this is just a stub. TODO: implement the internal GIS debuggign */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-07-09 10:47:42 +02:00
|
|
|
|
|
|
|
/**********************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_area : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_area(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_area s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_area() {}
|
|
|
|
virtual ~Create_func_area() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_as_wkb : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_as_wkb(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_as_wkb s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_as_wkb() {}
|
|
|
|
virtual ~Create_func_as_wkb() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_as_wkt : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_as_wkt(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_as_wkt s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_as_wkt() {}
|
|
|
|
virtual ~Create_func_as_wkt() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_centroid : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_centroid(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_centroid s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_centroid() {}
|
|
|
|
virtual ~Create_func_centroid() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_convexhull : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_convexhull(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_convexhull s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_convexhull() {}
|
|
|
|
virtual ~Create_func_convexhull() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_pointonsurface : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_pointonsurface(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_pointonsurface s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_pointonsurface() {}
|
|
|
|
virtual ~Create_func_pointonsurface() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_mbr_contains : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_mbr_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_CONTAINS_FUNC);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_mbr_contains s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_mbr_contains() {}
|
|
|
|
virtual ~Create_func_mbr_contains() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_contains : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_precise_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_CONTAINS_FUNC);
|
|
|
|
}
|
|
|
|
static Create_func_contains s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_contains() {}
|
|
|
|
virtual ~Create_func_contains() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_crosses : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_precise_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_CROSSES_FUNC);
|
|
|
|
}
|
|
|
|
static Create_func_crosses s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_crosses() {}
|
|
|
|
virtual ~Create_func_crosses() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_dimension : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_dimension(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_dimension s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_dimension() {}
|
|
|
|
virtual ~Create_func_dimension() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_mbr_disjoint : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_mbr_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_DISJOINT_FUNC);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_mbr_disjoint s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_mbr_disjoint() {}
|
|
|
|
virtual ~Create_func_mbr_disjoint() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_disjoint : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_precise_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_DISJOINT_FUNC);
|
|
|
|
}
|
|
|
|
static Create_func_disjoint s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_disjoint() {}
|
|
|
|
virtual ~Create_func_disjoint() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_distance : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item* create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_distance(thd, arg1, arg2);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_distance s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_distance() {}
|
|
|
|
virtual ~Create_func_distance() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_endpoint : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_decomp(thd, arg1,
|
|
|
|
Item_func::SP_ENDPOINT);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_endpoint s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_endpoint() {}
|
|
|
|
virtual ~Create_func_endpoint() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_envelope : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_envelope(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_envelope s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_envelope() {}
|
|
|
|
virtual ~Create_func_envelope() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
class Create_func_boundary : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_boundary(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_boundary s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_boundary() {}
|
|
|
|
virtual ~Create_func_boundary() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_mbr_equals : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_mbr_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_EQUALS_FUNC);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_mbr_equals s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_mbr_equals() {}
|
|
|
|
virtual ~Create_func_mbr_equals() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_equals : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_precise_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_EQUALS_FUNC);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_equals s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_equals() {}
|
|
|
|
virtual ~Create_func_equals() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_exteriorring : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_decomp(thd, arg1,
|
|
|
|
Item_func::SP_EXTERIORRING);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_exteriorring s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_exteriorring() {}
|
|
|
|
virtual ~Create_func_exteriorring() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_geometry_from_text : public Create_native_func
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_native(THD *thd, LEX_CSTRING *name, List<Item> *item_list);
|
|
|
|
|
|
|
|
static Create_func_geometry_from_text s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_geometry_from_text() {}
|
|
|
|
virtual ~Create_func_geometry_from_text() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
Item*
|
|
|
|
Create_func_geometry_from_text::create_native(THD *thd, LEX_CSTRING *name,
|
|
|
|
List<Item> *item_list)
|
|
|
|
{
|
|
|
|
Item *func= NULL;
|
|
|
|
int arg_count= 0;
|
|
|
|
|
|
|
|
if (item_list != NULL)
|
|
|
|
arg_count= item_list->elements;
|
|
|
|
|
|
|
|
switch (arg_count) {
|
|
|
|
case 1:
|
|
|
|
{
|
|
|
|
Item *param_1= item_list->pop();
|
|
|
|
func= new (thd->mem_root) Item_func_geometry_from_text(thd, param_1);
|
|
|
|
thd->lex->uncacheable(UNCACHEABLE_RAND);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2:
|
|
|
|
{
|
|
|
|
Item *param_1= item_list->pop();
|
|
|
|
Item *param_2= item_list->pop();
|
|
|
|
func= new (thd->mem_root) Item_func_geometry_from_text(thd, param_1, param_2);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
my_error(ER_WRONG_PARAMCOUNT_TO_NATIVE_FCT, MYF(0), name->str);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return func;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_geometry_from_wkb : public Create_native_func
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_native(THD *thd, LEX_CSTRING *name, List<Item> *item_list);
|
|
|
|
|
|
|
|
static Create_func_geometry_from_wkb s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_geometry_from_wkb() {}
|
|
|
|
virtual ~Create_func_geometry_from_wkb() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
Item*
|
|
|
|
Create_func_geometry_from_wkb::create_native(THD *thd, LEX_CSTRING *name,
|
|
|
|
List<Item> *item_list)
|
|
|
|
{
|
|
|
|
Item *func= NULL;
|
|
|
|
int arg_count= 0;
|
|
|
|
|
|
|
|
if (item_list != NULL)
|
|
|
|
arg_count= item_list->elements;
|
|
|
|
|
|
|
|
switch (arg_count) {
|
|
|
|
case 1:
|
|
|
|
{
|
|
|
|
Item *param_1= item_list->pop();
|
|
|
|
func= new (thd->mem_root) Item_func_geometry_from_wkb(thd, param_1);
|
|
|
|
thd->lex->uncacheable(UNCACHEABLE_RAND);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2:
|
|
|
|
{
|
|
|
|
Item *param_1= item_list->pop();
|
|
|
|
Item *param_2= item_list->pop();
|
|
|
|
func= new (thd->mem_root) Item_func_geometry_from_wkb(thd, param_1, param_2);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
my_error(ER_WRONG_PARAMCOUNT_TO_NATIVE_FCT, MYF(0), name->str);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return func;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_geometry_from_json : public Create_native_func
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_native(THD *thd, LEX_CSTRING *name, List<Item> *item_list);
|
|
|
|
|
|
|
|
static Create_func_geometry_from_json s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_geometry_from_json() {}
|
|
|
|
virtual ~Create_func_geometry_from_json() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
Item*
|
|
|
|
Create_func_geometry_from_json::create_native(THD *thd, LEX_CSTRING *name,
|
|
|
|
List<Item> *item_list)
|
|
|
|
{
|
|
|
|
Item *func= NULL;
|
|
|
|
int arg_count= 0;
|
|
|
|
|
|
|
|
if (item_list != NULL)
|
|
|
|
arg_count= item_list->elements;
|
|
|
|
|
|
|
|
switch (arg_count) {
|
|
|
|
case 1:
|
|
|
|
{
|
|
|
|
Item *json= item_list->pop();
|
|
|
|
func= new (thd->mem_root) Item_func_geometry_from_json(thd, json);
|
|
|
|
thd->lex->uncacheable(UNCACHEABLE_RAND);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2:
|
|
|
|
{
|
|
|
|
Item *json= item_list->pop();
|
|
|
|
Item *options= item_list->pop();
|
|
|
|
func= new (thd->mem_root) Item_func_geometry_from_json(thd, json, options);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 3:
|
|
|
|
{
|
|
|
|
Item *json= item_list->pop();
|
|
|
|
Item *options= item_list->pop();
|
|
|
|
Item *srid= item_list->pop();
|
|
|
|
func= new (thd->mem_root) Item_func_geometry_from_json(thd, json, options,
|
|
|
|
srid);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
my_error(ER_WRONG_PARAMCOUNT_TO_NATIVE_FCT, MYF(0), name->str);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return func;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_as_geojson : public Create_native_func
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_native(THD *thd, LEX_CSTRING *name, List<Item> *item_list);
|
|
|
|
|
|
|
|
static Create_func_as_geojson s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_as_geojson() {}
|
|
|
|
virtual ~Create_func_as_geojson() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
Item*
|
|
|
|
Create_func_as_geojson::create_native(THD *thd, LEX_CSTRING *name,
|
|
|
|
List<Item> *item_list)
|
|
|
|
{
|
|
|
|
Item *func= NULL;
|
|
|
|
int arg_count= 0;
|
|
|
|
|
|
|
|
if (item_list != NULL)
|
|
|
|
arg_count= item_list->elements;
|
|
|
|
|
|
|
|
switch (arg_count) {
|
|
|
|
case 1:
|
|
|
|
{
|
|
|
|
Item *geom= item_list->pop();
|
|
|
|
func= new (thd->mem_root) Item_func_as_geojson(thd, geom);
|
|
|
|
thd->lex->uncacheable(UNCACHEABLE_RAND);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2:
|
|
|
|
{
|
|
|
|
Item *geom= item_list->pop();
|
|
|
|
Item *max_dec= item_list->pop();
|
|
|
|
func= new (thd->mem_root) Item_func_as_geojson(thd, geom, max_dec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 3:
|
|
|
|
{
|
|
|
|
Item *geom= item_list->pop();
|
|
|
|
Item *max_dec= item_list->pop();
|
|
|
|
Item *options= item_list->pop();
|
|
|
|
func= new (thd->mem_root) Item_func_as_geojson(thd, geom, max_dec, options);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
my_error(ER_WRONG_PARAMCOUNT_TO_NATIVE_FCT, MYF(0), name->str);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return func;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_geometry_type : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_geometry_type(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_geometry_type s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_geometry_type() {}
|
|
|
|
virtual ~Create_func_geometry_type() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_geometryn : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_decomp_n(thd, arg1, arg2,
|
|
|
|
Item_func::SP_GEOMETRYN);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_geometryn s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_geometryn() {}
|
|
|
|
virtual ~Create_func_geometryn() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
#if !defined(DBUG_OFF)
|
|
|
|
class Create_func_gis_debug : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_gis_debug(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_gis_debug s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_gis_debug() {}
|
|
|
|
virtual ~Create_func_gis_debug() {}
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_glength : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_glength(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_glength s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_glength() {}
|
|
|
|
virtual ~Create_func_glength() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_interiorringn : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_decomp_n(thd, arg1, arg2,
|
|
|
|
Item_func::SP_INTERIORRINGN);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_interiorringn s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_interiorringn() {}
|
|
|
|
virtual ~Create_func_interiorringn() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_relate : public Create_func_arg3
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_3_arg(THD *thd, Item *arg1, Item *arg2, Item *arg3) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_relate(thd, arg1, arg2, arg3);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_relate s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_relate() {}
|
|
|
|
virtual ~Create_func_relate() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_mbr_intersects : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_mbr_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_INTERSECTS_FUNC);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_mbr_intersects s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_mbr_intersects() {}
|
|
|
|
virtual ~Create_func_mbr_intersects() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_intersects : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_precise_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_INTERSECTS_FUNC);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_intersects s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_intersects() {}
|
|
|
|
virtual ~Create_func_intersects() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_intersection : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item* create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_operation(thd, arg1, arg2,
|
|
|
|
Gcalc_function::op_intersection);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_intersection s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_intersection() {}
|
|
|
|
virtual ~Create_func_intersection() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_difference : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item* create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_operation(thd, arg1, arg2,
|
|
|
|
Gcalc_function::op_difference);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_difference s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_difference() {}
|
|
|
|
virtual ~Create_func_difference() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_union : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item* create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_operation(thd, arg1, arg2,
|
|
|
|
Gcalc_function::op_union);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_union s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_union() {}
|
|
|
|
virtual ~Create_func_union() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_symdifference : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item* create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_operation(thd, arg1, arg2,
|
|
|
|
Gcalc_function::op_symdifference);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_symdifference s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_symdifference() {}
|
|
|
|
virtual ~Create_func_symdifference() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_buffer : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item* create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_buffer(thd, arg1, arg2);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_buffer s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_buffer() {}
|
|
|
|
virtual ~Create_func_buffer() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_isclosed : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_isclosed(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_isclosed s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_isclosed() {}
|
|
|
|
virtual ~Create_func_isclosed() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_isring : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_isring(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_isring s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_isring() {}
|
|
|
|
virtual ~Create_func_isring() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_isempty : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_isempty(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_isempty s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_isempty() {}
|
|
|
|
virtual ~Create_func_isempty() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_issimple : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_issimple(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_issimple s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_issimple() {}
|
|
|
|
virtual ~Create_func_issimple() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_numgeometries : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_numgeometries(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_numgeometries s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_numgeometries() {}
|
|
|
|
virtual ~Create_func_numgeometries() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_numinteriorring : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_numinteriorring(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_numinteriorring s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_numinteriorring() {}
|
|
|
|
virtual ~Create_func_numinteriorring() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_numpoints : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_numpoints(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_numpoints s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_numpoints() {}
|
|
|
|
virtual ~Create_func_numpoints() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_mbr_overlaps : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_mbr_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_OVERLAPS_FUNC);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_mbr_overlaps s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_mbr_overlaps() {}
|
|
|
|
virtual ~Create_func_mbr_overlaps() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_overlaps : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_precise_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_OVERLAPS_FUNC);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_overlaps s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_overlaps() {}
|
|
|
|
virtual ~Create_func_overlaps() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_pointn : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_decomp_n(thd, arg1, arg2,
|
|
|
|
Item_func::SP_POINTN);
|
|
|
|
}
|
|
|
|
static Create_func_pointn s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_pointn() {}
|
|
|
|
virtual ~Create_func_pointn() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_srid : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_srid(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_srid s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_srid() {}
|
|
|
|
virtual ~Create_func_srid() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_startpoint : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_decomp(thd, arg1,
|
|
|
|
Item_func::SP_STARTPOINT);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_startpoint s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_startpoint() {}
|
|
|
|
virtual ~Create_func_startpoint() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_touches : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_precise_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_TOUCHES_FUNC);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_touches s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_touches() {}
|
|
|
|
virtual ~Create_func_touches() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_mbr_within : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_mbr_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_WITHIN_FUNC);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_mbr_within s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_mbr_within() {}
|
|
|
|
virtual ~Create_func_mbr_within() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_within : public Create_func_arg2
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_2_arg(THD *thd, Item *arg1, Item *arg2) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_spatial_precise_rel(thd, arg1, arg2,
|
|
|
|
Item_func::SP_WITHIN_FUNC);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_within s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_within() {}
|
|
|
|
virtual ~Create_func_within() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_x : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_x(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_x s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_x() {}
|
|
|
|
virtual ~Create_func_x() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Create_func_y : public Create_func_arg1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Item *create_1_arg(THD *thd, Item *arg1) override
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_y(thd, arg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Create_func_y s_singleton;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Create_func_y() {}
|
|
|
|
virtual ~Create_func_y() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*************************************************************************/
|
|
|
|
|
|
|
|
#if !defined(DBUG_OFF)
|
|
|
|
Create_func_gis_debug Create_func_gis_debug::s_singleton;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
Create_func_area Create_func_area::s_singleton;
|
|
|
|
Create_func_as_geojson Create_func_as_geojson::s_singleton;
|
|
|
|
Create_func_as_wkb Create_func_as_wkb::s_singleton;
|
|
|
|
Create_func_as_wkt Create_func_as_wkt::s_singleton;
|
|
|
|
Create_func_boundary Create_func_boundary::s_singleton;
|
|
|
|
Create_func_buffer Create_func_buffer::s_singleton;
|
|
|
|
Create_func_centroid Create_func_centroid::s_singleton;
|
|
|
|
Create_func_contains Create_func_contains::s_singleton;
|
|
|
|
Create_func_convexhull Create_func_convexhull::s_singleton;
|
|
|
|
Create_func_crosses Create_func_crosses::s_singleton;
|
|
|
|
Create_func_difference Create_func_difference::s_singleton;
|
|
|
|
Create_func_dimension Create_func_dimension::s_singleton;
|
|
|
|
Create_func_disjoint Create_func_disjoint::s_singleton;
|
|
|
|
Create_func_distance Create_func_distance::s_singleton;
|
|
|
|
Create_func_endpoint Create_func_endpoint::s_singleton;
|
|
|
|
Create_func_envelope Create_func_envelope::s_singleton;
|
|
|
|
Create_func_equals Create_func_equals::s_singleton;
|
|
|
|
Create_func_exteriorring Create_func_exteriorring::s_singleton;
|
|
|
|
Create_func_geometry_from_json Create_func_geometry_from_json::s_singleton;
|
|
|
|
Create_func_geometry_from_text Create_func_geometry_from_text::s_singleton;
|
|
|
|
Create_func_geometry_from_wkb Create_func_geometry_from_wkb::s_singleton;
|
|
|
|
Create_func_geometryn Create_func_geometryn::s_singleton;
|
|
|
|
Create_func_geometry_type Create_func_geometry_type::s_singleton;
|
|
|
|
Create_func_glength Create_func_glength::s_singleton;
|
|
|
|
Create_func_interiorringn Create_func_interiorringn::s_singleton;
|
|
|
|
Create_func_intersection Create_func_intersection::s_singleton;
|
|
|
|
Create_func_intersects Create_func_intersects::s_singleton;
|
|
|
|
Create_func_isclosed Create_func_isclosed::s_singleton;
|
|
|
|
Create_func_isempty Create_func_isempty::s_singleton;
|
|
|
|
Create_func_isring Create_func_isring::s_singleton;
|
|
|
|
Create_func_issimple Create_func_issimple::s_singleton;
|
|
|
|
Create_func_mbr_contains Create_func_mbr_contains::s_singleton;
|
|
|
|
Create_func_mbr_disjoint Create_func_mbr_disjoint::s_singleton;
|
|
|
|
Create_func_mbr_equals Create_func_mbr_equals::s_singleton;
|
|
|
|
Create_func_mbr_intersects Create_func_mbr_intersects::s_singleton;
|
|
|
|
Create_func_mbr_overlaps Create_func_mbr_overlaps::s_singleton;
|
|
|
|
Create_func_mbr_within Create_func_mbr_within::s_singleton;
|
|
|
|
Create_func_numgeometries Create_func_numgeometries::s_singleton;
|
|
|
|
Create_func_numinteriorring Create_func_numinteriorring::s_singleton;
|
|
|
|
Create_func_numpoints Create_func_numpoints::s_singleton;
|
|
|
|
Create_func_overlaps Create_func_overlaps::s_singleton;
|
|
|
|
Create_func_pointn Create_func_pointn::s_singleton;
|
|
|
|
Create_func_pointonsurface Create_func_pointonsurface::s_singleton;
|
|
|
|
Create_func_relate Create_func_relate::s_singleton;
|
|
|
|
Create_func_srid Create_func_srid::s_singleton;
|
|
|
|
Create_func_startpoint Create_func_startpoint::s_singleton;
|
|
|
|
Create_func_symdifference Create_func_symdifference::s_singleton;
|
|
|
|
Create_func_touches Create_func_touches::s_singleton;
|
|
|
|
Create_func_union Create_func_union::s_singleton;
|
|
|
|
Create_func_within Create_func_within::s_singleton;
|
|
|
|
Create_func_x Create_func_x::s_singleton;
|
|
|
|
Create_func_y Create_func_y::s_singleton;
|
|
|
|
|
|
|
|
/*************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
#define GEOM_BUILDER(F) & F::s_singleton
|
|
|
|
|
|
|
|
|
|
|
|
static Native_func_registry func_array_geom[] =
|
|
|
|
{
|
|
|
|
#ifndef DBUG_OFF
|
|
|
|
{ { STRING_WITH_LEN("ST_GIS_DEBUG") }, GEOM_BUILDER(Create_func_gis_debug)},
|
|
|
|
#endif
|
|
|
|
{ { STRING_WITH_LEN("AREA") }, GEOM_BUILDER(Create_func_area)},
|
|
|
|
{ { STRING_WITH_LEN("ASBINARY") }, GEOM_BUILDER(Create_func_as_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ASTEXT") }, GEOM_BUILDER(Create_func_as_wkt)},
|
|
|
|
{ { STRING_WITH_LEN("ASWKB") }, GEOM_BUILDER(Create_func_as_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ASWKT") }, GEOM_BUILDER(Create_func_as_wkt)},
|
|
|
|
{ { STRING_WITH_LEN("BOUNDARY") }, GEOM_BUILDER(Create_func_boundary)},
|
|
|
|
{ { STRING_WITH_LEN("BUFFER") }, GEOM_BUILDER(Create_func_buffer)},
|
|
|
|
{ { STRING_WITH_LEN("CENTROID") }, GEOM_BUILDER(Create_func_centroid)},
|
|
|
|
{ { STRING_WITH_LEN("CONTAINS") }, GEOM_BUILDER(Create_func_contains)},
|
|
|
|
{ { STRING_WITH_LEN("CONVEXHULL") }, GEOM_BUILDER(Create_func_convexhull)},
|
|
|
|
{ { STRING_WITH_LEN("CROSSES") }, GEOM_BUILDER(Create_func_crosses)},
|
|
|
|
{ { STRING_WITH_LEN("DIMENSION") }, GEOM_BUILDER(Create_func_dimension)},
|
|
|
|
{ { STRING_WITH_LEN("DISJOINT") }, GEOM_BUILDER(Create_func_mbr_disjoint)},
|
|
|
|
{ { STRING_WITH_LEN("ENDPOINT") }, GEOM_BUILDER(Create_func_endpoint)},
|
|
|
|
{ { STRING_WITH_LEN("ENVELOPE") }, GEOM_BUILDER(Create_func_envelope)},
|
|
|
|
{ { STRING_WITH_LEN("EQUALS") }, GEOM_BUILDER(Create_func_equals)},
|
|
|
|
{ { STRING_WITH_LEN("EXTERIORRING") }, GEOM_BUILDER(Create_func_exteriorring)},
|
|
|
|
{ { STRING_WITH_LEN("GEOMCOLLFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("GEOMCOLLFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("GEOMETRYCOLLECTIONFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("GEOMETRYCOLLECTIONFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("GEOMETRYFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("GEOMETRYFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("GEOMETRYN") }, GEOM_BUILDER(Create_func_geometryn)},
|
|
|
|
{ { STRING_WITH_LEN("GEOMETRYTYPE") }, GEOM_BUILDER(Create_func_geometry_type)},
|
|
|
|
{ { STRING_WITH_LEN("GEOMFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("GEOMFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("GLENGTH") }, GEOM_BUILDER(Create_func_glength)},
|
|
|
|
{ { STRING_WITH_LEN("INTERIORRINGN") }, GEOM_BUILDER(Create_func_interiorringn)},
|
|
|
|
{ { STRING_WITH_LEN("INTERSECTS") }, GEOM_BUILDER(Create_func_mbr_intersects)},
|
|
|
|
{ { STRING_WITH_LEN("ISCLOSED") }, GEOM_BUILDER(Create_func_isclosed)},
|
|
|
|
{ { STRING_WITH_LEN("ISEMPTY") }, GEOM_BUILDER(Create_func_isempty)},
|
|
|
|
{ { STRING_WITH_LEN("ISRING") }, GEOM_BUILDER(Create_func_isring)},
|
|
|
|
{ { STRING_WITH_LEN("ISSIMPLE") }, GEOM_BUILDER(Create_func_issimple)},
|
|
|
|
{ { STRING_WITH_LEN("LINEFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("LINEFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("LINESTRINGFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("LINESTRINGFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("MBRCONTAINS") }, GEOM_BUILDER(Create_func_mbr_contains)},
|
|
|
|
{ { STRING_WITH_LEN("MBRDISJOINT") }, GEOM_BUILDER(Create_func_mbr_disjoint)},
|
|
|
|
{ { STRING_WITH_LEN("MBREQUAL") }, GEOM_BUILDER(Create_func_mbr_equals)},
|
|
|
|
{ { STRING_WITH_LEN("MBREQUALS") }, GEOM_BUILDER(Create_func_mbr_equals)},
|
|
|
|
{ { STRING_WITH_LEN("MBRINTERSECTS") }, GEOM_BUILDER(Create_func_mbr_intersects)},
|
|
|
|
{ { STRING_WITH_LEN("MBROVERLAPS") }, GEOM_BUILDER(Create_func_mbr_overlaps)},
|
|
|
|
{ { STRING_WITH_LEN("MBRTOUCHES") }, GEOM_BUILDER(Create_func_touches)},
|
|
|
|
{ { STRING_WITH_LEN("MBRWITHIN") }, GEOM_BUILDER(Create_func_mbr_within)},
|
|
|
|
{ { STRING_WITH_LEN("MLINEFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("MLINEFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("MPOINTFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("MPOINTFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("MPOLYFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("MPOLYFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("MULTILINESTRINGFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("MULTILINESTRINGFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("MULTIPOINTFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("MULTIPOINTFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("MULTIPOLYGONFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("MULTIPOLYGONFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("NUMGEOMETRIES") }, GEOM_BUILDER(Create_func_numgeometries)},
|
|
|
|
{ { STRING_WITH_LEN("NUMINTERIORRINGS") }, GEOM_BUILDER(Create_func_numinteriorring)},
|
|
|
|
{ { STRING_WITH_LEN("NUMPOINTS") }, GEOM_BUILDER(Create_func_numpoints)},
|
|
|
|
{ { STRING_WITH_LEN("OVERLAPS") }, GEOM_BUILDER(Create_func_mbr_overlaps)},
|
|
|
|
{ { STRING_WITH_LEN("POINTFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("POINTFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("POINTN") }, GEOM_BUILDER(Create_func_pointn)},
|
|
|
|
{ { STRING_WITH_LEN("POINTONSURFACE") }, GEOM_BUILDER(Create_func_pointonsurface)},
|
|
|
|
{ { STRING_WITH_LEN("POLYFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("POLYFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("POLYGONFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("POLYGONFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("SRID") }, GEOM_BUILDER(Create_func_srid)},
|
|
|
|
{ { STRING_WITH_LEN("ST_AREA") }, GEOM_BUILDER(Create_func_area)},
|
|
|
|
{ { STRING_WITH_LEN("STARTPOINT") }, GEOM_BUILDER(Create_func_startpoint)},
|
|
|
|
{ { STRING_WITH_LEN("ST_ASBINARY") }, GEOM_BUILDER(Create_func_as_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_ASGEOJSON") }, GEOM_BUILDER(Create_func_as_geojson)},
|
|
|
|
{ { STRING_WITH_LEN("ST_ASTEXT") }, GEOM_BUILDER(Create_func_as_wkt)},
|
|
|
|
{ { STRING_WITH_LEN("ST_ASWKB") }, GEOM_BUILDER(Create_func_as_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_ASWKT") }, GEOM_BUILDER(Create_func_as_wkt)},
|
|
|
|
{ { STRING_WITH_LEN("ST_BOUNDARY") }, GEOM_BUILDER(Create_func_boundary)},
|
|
|
|
{ { STRING_WITH_LEN("ST_BUFFER") }, GEOM_BUILDER(Create_func_buffer)},
|
|
|
|
{ { STRING_WITH_LEN("ST_CENTROID") }, GEOM_BUILDER(Create_func_centroid)},
|
|
|
|
{ { STRING_WITH_LEN("ST_CONTAINS") }, GEOM_BUILDER(Create_func_contains)},
|
|
|
|
{ { STRING_WITH_LEN("ST_CONVEXHULL") }, GEOM_BUILDER(Create_func_convexhull)},
|
|
|
|
{ { STRING_WITH_LEN("ST_CROSSES") }, GEOM_BUILDER(Create_func_crosses)},
|
|
|
|
{ { STRING_WITH_LEN("ST_DIFFERENCE") }, GEOM_BUILDER(Create_func_difference)},
|
|
|
|
{ { STRING_WITH_LEN("ST_DIMENSION") }, GEOM_BUILDER(Create_func_dimension)},
|
|
|
|
{ { STRING_WITH_LEN("ST_DISJOINT") }, GEOM_BUILDER(Create_func_disjoint)},
|
|
|
|
{ { STRING_WITH_LEN("ST_DISTANCE") }, GEOM_BUILDER(Create_func_distance)},
|
|
|
|
{ { STRING_WITH_LEN("ST_ENDPOINT") }, GEOM_BUILDER(Create_func_endpoint)},
|
|
|
|
{ { STRING_WITH_LEN("ST_ENVELOPE") }, GEOM_BUILDER(Create_func_envelope)},
|
|
|
|
{ { STRING_WITH_LEN("ST_EQUALS") }, GEOM_BUILDER(Create_func_equals)},
|
|
|
|
{ { STRING_WITH_LEN("ST_EQUALS") }, GEOM_BUILDER(Create_func_equals)},
|
|
|
|
{ { STRING_WITH_LEN("ST_EXTERIORRING") }, GEOM_BUILDER(Create_func_exteriorring)},
|
|
|
|
{ { STRING_WITH_LEN("ST_GEOMCOLLFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_GEOMCOLLFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_GEOMETRYCOLLECTIONFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_GEOMETRYCOLLECTIONFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_GEOMETRYFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_GEOMETRYFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_GEOMETRYN") }, GEOM_BUILDER(Create_func_geometryn)},
|
|
|
|
{ { STRING_WITH_LEN("ST_GEOMETRYTYPE") }, GEOM_BUILDER(Create_func_geometry_type)},
|
|
|
|
{ { STRING_WITH_LEN("ST_GEOMFROMGEOJSON") }, GEOM_BUILDER(Create_func_geometry_from_json)},
|
|
|
|
{ { STRING_WITH_LEN("ST_GEOMFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_GEOMFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_INTERIORRINGN") }, GEOM_BUILDER(Create_func_interiorringn)},
|
|
|
|
{ { STRING_WITH_LEN("ST_INTERSECTION") }, GEOM_BUILDER(Create_func_intersection)},
|
|
|
|
{ { STRING_WITH_LEN("ST_INTERSECTS") }, GEOM_BUILDER(Create_func_intersects)},
|
|
|
|
{ { STRING_WITH_LEN("ST_ISCLOSED") }, GEOM_BUILDER(Create_func_isclosed)},
|
|
|
|
{ { STRING_WITH_LEN("ST_ISEMPTY") }, GEOM_BUILDER(Create_func_isempty)},
|
|
|
|
{ { STRING_WITH_LEN("ST_ISRING") }, GEOM_BUILDER(Create_func_isring)},
|
|
|
|
{ { STRING_WITH_LEN("ST_ISSIMPLE") }, GEOM_BUILDER(Create_func_issimple)},
|
|
|
|
{ { STRING_WITH_LEN("ST_LENGTH") }, GEOM_BUILDER(Create_func_glength)},
|
|
|
|
{ { STRING_WITH_LEN("ST_LINEFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_LINEFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_LINESTRINGFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_LINESTRINGFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_MLINEFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_MLINEFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_MPOINTFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_MPOINTFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_MPOLYFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_MPOLYFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_MULTILINESTRINGFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_MULTILINESTRINGFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_MULTIPOINTFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_MULTIPOINTFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_MULTIPOLYGONFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_MULTIPOLYGONFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_NUMGEOMETRIES") }, GEOM_BUILDER(Create_func_numgeometries)},
|
|
|
|
{ { STRING_WITH_LEN("ST_NUMINTERIORRINGS") }, GEOM_BUILDER(Create_func_numinteriorring)},
|
|
|
|
{ { STRING_WITH_LEN("ST_NUMPOINTS") }, GEOM_BUILDER(Create_func_numpoints)},
|
|
|
|
{ { STRING_WITH_LEN("ST_OVERLAPS") }, GEOM_BUILDER(Create_func_overlaps)},
|
|
|
|
{ { STRING_WITH_LEN("ST_POINTFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_POINTFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_POINTN") }, GEOM_BUILDER(Create_func_pointn)},
|
|
|
|
{ { STRING_WITH_LEN("ST_POINTONSURFACE") }, GEOM_BUILDER(Create_func_pointonsurface)},
|
|
|
|
{ { STRING_WITH_LEN("ST_POLYFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_POLYFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_POLYGONFROMTEXT") }, GEOM_BUILDER(Create_func_geometry_from_text)},
|
|
|
|
{ { STRING_WITH_LEN("ST_POLYGONFROMWKB") }, GEOM_BUILDER(Create_func_geometry_from_wkb)},
|
|
|
|
{ { STRING_WITH_LEN("ST_RELATE") }, GEOM_BUILDER(Create_func_relate)},
|
|
|
|
{ { STRING_WITH_LEN("ST_SRID") }, GEOM_BUILDER(Create_func_srid)},
|
|
|
|
{ { STRING_WITH_LEN("ST_STARTPOINT") }, GEOM_BUILDER(Create_func_startpoint)},
|
|
|
|
{ { STRING_WITH_LEN("ST_SYMDIFFERENCE") }, GEOM_BUILDER(Create_func_symdifference)},
|
|
|
|
{ { STRING_WITH_LEN("ST_TOUCHES") }, GEOM_BUILDER(Create_func_touches)},
|
|
|
|
{ { STRING_WITH_LEN("ST_UNION") }, GEOM_BUILDER(Create_func_union)},
|
|
|
|
{ { STRING_WITH_LEN("ST_WITHIN") }, GEOM_BUILDER(Create_func_within)},
|
|
|
|
{ { STRING_WITH_LEN("ST_X") }, GEOM_BUILDER(Create_func_x)},
|
|
|
|
{ { STRING_WITH_LEN("ST_Y") }, GEOM_BUILDER(Create_func_y)},
|
|
|
|
{ { STRING_WITH_LEN("TOUCHES") }, GEOM_BUILDER(Create_func_touches)},
|
|
|
|
{ { STRING_WITH_LEN("WITHIN") }, GEOM_BUILDER(Create_func_within)},
|
|
|
|
{ { STRING_WITH_LEN("X") }, GEOM_BUILDER(Create_func_x)},
|
|
|
|
{ { STRING_WITH_LEN("Y") }, GEOM_BUILDER(Create_func_y)},
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2019-10-16 14:22:04 +02:00
|
|
|
Native_func_registry_array
|
|
|
|
native_func_registry_array_geom(func_array_geom,
|
|
|
|
array_elements(func_array_geom));
|
2019-07-09 10:47:42 +02:00
|
|
|
|
2004-01-15 18:06:22 +01:00
|
|
|
#endif /*HAVE_SPATIAL*/
|