summaryrefslogtreecommitdiff
path: root/sql/unireg.cc
diff options
context:
space:
mode:
authorSergei Golubchik <serg@mariadb.org>2016-11-07 17:17:40 +0100
committerSergei Golubchik <serg@mariadb.org>2016-12-12 20:35:41 +0100
commita411d7f4f670c24b43b50f7d2a1129e10218f4a7 (patch)
treecc1e9cdf6b85bdcec8c0369e502d4d211b2f7263 /sql/unireg.cc
parent8b3b6dc377c548b1b72978a015af999cf6e99760 (diff)
downloadmariadb-git-a411d7f4f670c24b43b50f7d2a1129e10218f4a7.tar.gz
store/show vcols as item->print()
otherwise we'd need to store sql_mode *per vcol* (consider CREATE INDEX...) and how SHOW CREATE TABLE would support that? Additionally, get rid of vcol::expr_str, just to make sure the string is always generated and never leaked in the original form.
Diffstat (limited to 'sql/unireg.cc')
-rw-r--r--sql/unireg.cc262
1 files changed, 96 insertions, 166 deletions
diff --git a/sql/unireg.cc b/sql/unireg.cc
index 66fde186b5b..c102835a7bb 100644
--- a/sql/unireg.cc
+++ b/sql/unireg.cc
@@ -42,13 +42,11 @@
static uint pack_keys(uchar *,uint, KEY *, ulong);
static bool pack_header(THD *, uchar *, List<Create_field> &, HA_CREATE_INFO *,
ulong, handler *);
+static bool pack_vcols(String *, List<Create_field> &, List<Virtual_column_info> *);
static uint get_interval_id(uint *,List<Create_field> &, Create_field *);
static bool pack_fields(uchar **, List<Create_field> &, HA_CREATE_INFO*,
ulong);
-static void pack_constraints(uchar **buff, List<Virtual_column_info> *constr);
static size_t packed_fields_length(List<Create_field> &);
-static size_t packed_constraints_length(THD *, HA_CREATE_INFO*,
- List<Virtual_column_info> *);
static bool make_empty_rec(THD *, uchar *, uint, List<Create_field> &, uint,
ulong);
@@ -120,6 +118,7 @@ LEX_CUSTRING build_frm_image(THD *thd, const char *table,
int error;
uchar *frm_ptr, *pos;
LEX_CUSTRING frm= {0,0};
+ StringBuffer<MAX_FIELD_WIDTH> vcols;
DBUG_ENTER("build_frm_image");
/* If fixed row records, we need one bit to check for deleted rows */
@@ -127,9 +126,19 @@ LEX_CUSTRING build_frm_image(THD *thd, const char *table,
create_info->null_bits++;
data_offset= (create_info->null_bits + 7) / 8;
+ sql_mode_t save_sql_mode= thd->variables.sql_mode;
+ thd->variables.sql_mode &= ~MODE_ANSI_QUOTES;
+ error= pack_vcols(&vcols, create_fields, create_info->check_constraint_list);
+ thd->variables.sql_mode= save_sql_mode;
+
+ if (error)
+ DBUG_RETURN(frm);
+
+ if (vcols.length())
+ create_info->expression_length= vcols.length() + FRM_VCOL_NEW_BASE_SIZE;
+
error= pack_header(thd, forminfo, create_fields, create_info,
data_offset, db_file);
-
if (error)
DBUG_RETURN(frm);
@@ -336,6 +345,15 @@ LEX_CUSTRING build_frm_image(THD *thd, const char *table,
if (pack_fields(&pos, create_fields, create_info, data_offset))
goto err;
+ if (vcols.length())
+ {
+ /* Store header for packed fields (extra space for future) */
+ bzero(pos, FRM_VCOL_NEW_BASE_SIZE);
+ pos+= FRM_VCOL_NEW_BASE_SIZE;
+ memcpy(pos, vcols.ptr(), vcols.length());
+ pos+= vcols.length();
+ }
+
{
/*
Restore all UCS2 intervals.
@@ -496,92 +514,73 @@ static uint pack_keys(uchar *keybuff, uint key_count, KEY *keyinfo,
/**
- Calculate and check length of stored expression (virtual, def, check)
-
- Convert string to utf8, if it isn't already
+ Pack the expression (for GENERATED ALWAYS AS, DEFAULT, CHECK)
- @param thd Thread handler. Used for memory allocation
- @param v_col Virtual expression. Can be 0
- @param CREATE INFO For characterset
- @param length Sum total lengths here
-
- Note to make calls easier, one can call this with v_col == 0
+ The data is stored as:
+ 1 byte type (enum_vcol_info_type)
+ 2 bytes field_number
+ 2 bytes length of expression
+ 1 byte length of name
+ name
+ next bytes column expression (text data)
@return 0 ok
@return 1 error (out of memory or wrong characters in expression)
*/
-static bool add_expr_length(THD *thd, Virtual_column_info **v_col_ptr,
- size_t *length)
+static bool pack_expression(String *buf, Virtual_column_info *vcol,
+ uint field_nr, enum_vcol_info_type type)
{
- Virtual_column_info *v_col= *v_col_ptr;
- if (!v_col)
- return 0;
-
- /*
- Convert string to utf8 for storage.
- */
- if (!v_col->utf8)
- {
- /*
- This v_col comes from the parser (e.g. CREATE TABLE) or
- from old (before 10.2.1) frm.
-
- We have to create a new Virtual_column_info as for alter table,
- the current one may be shared with the original table.
- */
- Virtual_column_info *new_vcol= new (thd->mem_root) Virtual_column_info();
- LEX_STRING to;
- if (thd->copy_with_error(&my_charset_utf8mb4_general_ci,
- &to,
- thd->variables.character_set_client,
- v_col->expr_str.str, v_col->expr_str.length))
- return 1;
- *new_vcol= *v_col;
- new_vcol->expr_str= to;
- new_vcol->utf8= 1;
- *v_col_ptr= new_vcol;
- v_col= new_vcol;
- }
-
- /*
- Sum up the length of the expression string, it's optional name
- and the header.
- */
- (*length)+= (FRM_VCOL_NEW_HEADER_SIZE + v_col->name.length +
- v_col->expr_str.length);
+ if (buf->reserve(FRM_VCOL_NEW_HEADER_SIZE + vcol->name.length))
+ return 1;
+
+ buf->q_append((char) type);
+ buf->q_append2b(field_nr);
+ size_t len_off= buf->length();
+ buf->q_append2b(0); // to be added later
+ buf->q_append((char)vcol->name.length);
+ buf->q_append(vcol->name.str, vcol->name.length);
+ size_t expr_start= buf->length();
+ vcol->print(buf);
+ size_t expr_len= buf->length() - expr_start;
+ if (expr_len >= 65536)
+ {
+ my_error(ER_EXPRESSION_IS_TOO_BIG, MYF(0), vcol_type_name(type));
+ return 1;
+ }
+ int2store(buf->ptr() + len_off, expr_len);
return 0;
}
-/*
- pack_expression
-
- The data is stored as:
- 1 byte type (0 virtual, 1 virtual stored, 2 def, 3 check)
- 2 bytes field_number
- 2 bytes length of expression
- 1 byte length of name
- name
- next bytes column expression (text data)
-*/
-
-static void pack_expression(uchar **buff, Virtual_column_info *vcol,
- uint offset, uint type)
+static bool pack_vcols(String *buf, List<Create_field> &create_fields,
+ List<Virtual_column_info> *check_constraint_list)
{
- (*buff)[0]= (uchar) type;
- int2store((*buff)+1, offset);
- /*
- expr_str.length < 64K as we have checked that the total size of the
- frm file is < 64K
- */
- int2store((*buff)+3, vcol->expr_str.length);
- (*buff)[5]= vcol->name.length;
- (*buff)+= FRM_VCOL_NEW_HEADER_SIZE;
- memcpy((*buff), vcol->name.str, vcol->name.length);
- (*buff)+= vcol->name.length;
- memcpy((*buff), vcol->expr_str.str, vcol->expr_str.length);
- (*buff)+= vcol->expr_str.length;
+ List_iterator<Create_field> it(create_fields);
+ Create_field *field;
+
+ for (uint field_nr=0; (field= it++); field_nr++)
+ {
+ if (field->vcol_info)
+ if (pack_expression(buf, field->vcol_info, field_nr,
+ field->vcol_info->stored_in_db
+ ? VCOL_GENERATED_STORED : VCOL_GENERATED_VIRTUAL))
+ return 1;
+ if (field->has_default_expression() && !field->has_default_now_unireg_check())
+ if (pack_expression(buf, field->default_value, field_nr, VCOL_DEFAULT))
+ return 1;
+ if (field->check_constraint)
+ if (pack_expression(buf, field->check_constraint, field_nr,
+ VCOL_CHECK_FIELD))
+ return 1;
+ }
+
+ List_iterator<Virtual_column_info> cit(*check_constraint_list);
+ Virtual_column_info *check;
+ while ((check= cit++))
+ if (pack_expression(buf, check, UINT_MAX32, VCOL_CHECK_TABLE))
+ return 1;
+ return 0;
}
@@ -592,10 +591,10 @@ static bool pack_header(THD *thd, uchar *forminfo,
HA_CREATE_INFO *create_info, ulong data_offset,
handler *file)
{
- uint length,int_count,int_length, int_parts;
+ uint int_count,int_length, int_parts;
uint time_stamp_pos,null_fields;
uint table_options= create_info->table_options;
- size_t reclength, totlength, n_length, com_length, expression_length;
+ size_t length, reclength, totlength, n_length, com_length;
DBUG_ENTER("pack_header");
if (create_fields.elements > MAX_FIELDS)
@@ -611,16 +610,6 @@ static bool pack_header(THD *thd, uchar *forminfo,
n_length=2L;
create_info->field_check_constraints= 0;
- if (create_info->check_constraint_list->elements)
- {
- expression_length= packed_constraints_length(thd, create_info,
- create_info->check_constraint_list);
- if (!expression_length)
- DBUG_RETURN(1); // Wrong characterset
- }
- else
- expression_length= 0;
-
/* Check fields */
List_iterator<Create_field> it(create_fields);
Create_field *field;
@@ -630,14 +619,6 @@ static bool pack_header(THD *thd, uchar *forminfo,
ER_TOO_LONG_FIELD_COMMENT, field->field_name))
DBUG_RETURN(1);
- if (add_expr_length(thd, &field->vcol_info, &expression_length))
- DBUG_RETURN(1);
- if (field->default_value && field->default_value->expr_str.length)
- if (add_expr_length(thd, &field->default_value, &expression_length))
- DBUG_RETURN(1);
- if (add_expr_length(thd, &field->check_constraint, &expression_length))
- DBUG_RETURN(1);
-
totlength+= field->length;
com_length+= field->comment.length;
/*
@@ -714,31 +695,22 @@ static bool pack_header(THD *thd, uchar *forminfo,
DBUG_RETURN(1);
}
- if (expression_length)
- {
- expression_length+= FRM_VCOL_NEW_BASE_SIZE;
- create_info->expression_length= expression_length;
- }
-
/* Hack to avoid bugs with small static rows in MySQL */
- reclength=MY_MAX(file->min_record_length(table_options),reclength);
- if ((ulong) create_fields.elements*FCOMP+FRM_FORMINFO_SIZE+
- n_length+int_length+com_length+expression_length > 65535L ||
- int_count > 255)
+ reclength= MY_MAX(file->min_record_length(table_options), reclength);
+ length= n_length + create_fields.elements*FCOMP + FRM_FORMINFO_SIZE +
+ int_length + com_length + create_info->expression_length;
+ if (length > 65535L || int_count > 255)
{
my_message(ER_TOO_MANY_FIELDS, "Table definition is too large", MYF(0));
DBUG_RETURN(1);
}
bzero((char*)forminfo,FRM_FORMINFO_SIZE);
- length=(create_fields.elements*FCOMP+FRM_FORMINFO_SIZE+n_length+int_length+
- com_length+expression_length);
int2store(forminfo,length);
- forminfo[256] = 0;
int2store(forminfo+258,create_fields.elements);
- int2store(forminfo+260,0); // Screen length, not used anymore
+ // bytes 260-261 are unused
int2store(forminfo+262,totlength);
- int2store(forminfo+264,0); // unused
+ // bytes 264-265 are unused
int2store(forminfo+266,reclength);
int2store(forminfo+268,n_length);
int2store(forminfo+270,int_count);
@@ -749,7 +721,7 @@ static bool pack_header(THD *thd, uchar *forminfo,
int2store(forminfo+280,22); /* Rows needed */
int2store(forminfo+282,null_fields);
int2store(forminfo+284,com_length);
- int2store(forminfo+286,expression_length);
+ int2store(forminfo+286,create_info->expression_length);
DBUG_RETURN(0);
} /* pack_header */
@@ -811,29 +783,6 @@ static size_t packed_fields_length(List<Create_field> &create_fields)
DBUG_RETURN(length);
}
-
-static size_t packed_constraints_length(THD *thd, HA_CREATE_INFO *info,
- List<Virtual_column_info> *constr)
-{
- List_iterator<Virtual_column_info> it(*constr);
- size_t length= 0;
- Virtual_column_info *check;
-
- while ((check= it++))
- if (add_expr_length(thd, it.ref(), &length))
- return 0;
- return length;
-}
-
-static void pack_constraints(uchar **buff, List<Virtual_column_info> *constr)
-{
- List_iterator<Virtual_column_info> it(*constr);
- Virtual_column_info *check;
- while ((check= it++))
- pack_expression(buff, check, UINT_MAX32, 4);
-}
-
-
/* Save fields, fieldnames and intervals */
static bool pack_fields(uchar **buff_arg, List<Create_field> &create_fields,
@@ -959,27 +908,6 @@ static bool pack_fields(uchar **buff_arg, List<Create_field> &create_fields,
buff+= field->comment.length;
}
}
-
- if (create_info->expression_length)
- {
- /* Store header for packed fields (extra space for future) */
- bzero(buff, FRM_VCOL_NEW_BASE_SIZE);
- buff+= FRM_VCOL_NEW_BASE_SIZE;
-
- /* Store expressions */
- it.rewind();
- for (uint field_nr=0 ; (field= it++) ; field_nr++)
- {
- if (field->vcol_info)
- pack_expression(&buff, field->vcol_info, field_nr,
- field->vcol_info->stored_in_db ? 1 : 0);
- if (field->default_value && field->default_value->expr_str.length)
- pack_expression(&buff, field->default_value, field_nr, 2);
- if (field->check_constraint)
- pack_expression(&buff, field->check_constraint, field_nr, 3);
- }
- pack_constraints(&buff, create_info->check_constraint_list);
- }
*buff_arg= buff;
DBUG_RETURN(0);
}
@@ -1028,12 +956,9 @@ static bool make_empty_rec(THD *thd, uchar *buff, uint table_options,
field->sql_type,
field->charset,
field->geom_type, field->srid,
- field->unireg_check == Field::TIMESTAMP_DNUN_FIELD
- ? Field::TIMESTAMP_UN_FIELD
- : field->unireg_check == Field::TIMESTAMP_DN_FIELD
- ? Field::NONE : field->unireg_check,
- field->save_interval ? field->save_interval :
- field->interval,
+ field->unireg_check,
+ field->save_interval ? field->save_interval
+ : field->interval,
field->field_name);
if (!regfield)
{
@@ -1053,9 +978,14 @@ static bool make_empty_rec(THD *thd, uchar *buff, uint table_options,
if (field->sql_type == MYSQL_TYPE_BIT && !f_bit_as_char(field->pack_flag))
null_count+= field->length & 7;
- if (field->default_value && !field->has_default_expression())
+ if (field->default_value && !field->default_value->flags &&
+ !(field->flags & BLOB_FLAG))
{
- int res= field->default_value->expr_item->save_in_field(regfield, 1);
+ Item *expr= field->default_value->expr_item;
+ int res= !expr->fixed && // may be already fixed if ALTER TABLE
+ expr->fix_fields(thd, &expr);
+ if (!res)
+ res= expr->save_in_field(regfield, 1);
/* If not ok or warning of level 'note' */
if (res != 0 && res != 3)
{