Commit e4f6fd5e authored by Alexander Barkov's avatar Alexander Barkov

MDEV-10743 LDML: a new syntax to reuse sort order from another 8bit simple collation

parent 8ae65920
...@@ -319,8 +319,29 @@ ...@@ -319,8 +319,29 @@
</charset> </charset>
<charset name="ascii2"> <charset name="ascii2">
<!--
Notes:
- ascii2 has two collations with "binary" flag.
ctype_ldml.test makes sure that
CREATE TABLE t1 (a VARCHAR(10) CHARACTER SET ascii2 BINARY);
uses ascii2_bin2, which is the collation with the least ID.
- ascii2_general_inherited_ci inherits sort order in ascii2.xml
- ascii2_genegal_inherited2_ci inherits sort order directly in this file.
-->
<collation name="ascii2_bin2" id="319" flag="binary"/>
<collation name="ascii2_general_ci" id="320" flag="primary"/> <collation name="ascii2_general_ci" id="320" flag="primary"/>
<collation name="ascii2_bin" id="321" flag="binary"/> <collation name="ascii2_bin" id="321" flag="binary"/>
<collation name="ascii2_general_inherited_ci" id="322"/>
<collation name="ascii2_general_inherited2_ci" id="323">
<rules>
<import source="ascii2_general_ci"/>
</rules>
</collation>
<collation name="ascii2_badly_inherited_ci" id="324">
<rules>
<import source="ascii2_non_existing_ci"/>
</rules>
</collation>
</charset> </charset>
<charset name="latin1"> <charset name="latin1">
......
...@@ -116,6 +116,12 @@ ...@@ -116,6 +116,12 @@
<collation name="ascii2_bin" flag="binary"/> <collation name="ascii2_bin" flag="binary"/>
<collation name="ascii2_general_inherited_ci">
<rules>
<import source="ascii2_general_ci"/>
</rules>
</collation>
</charset> </charset>
</charsets> </charsets>
...@@ -413,16 +413,48 @@ DROP TABLE t1; ...@@ -413,16 +413,48 @@ DROP TABLE t1;
--echo # --echo #
--echo # Testing that the MY_CS_PUREASCII flag is set properly --echo # Testing that the MY_CS_PUREASCII flag is set properly
--echo # Comparison between ascii2 and latin1 should not give "illegal collation error"
--echo # --echo #
CREATE TABLE t1 (a VARCHAR(10) CHARACTER SET ascii2, b VARCHAR(10) CHARACTER SET latin1); CREATE TABLE t1 (a VARCHAR(10) CHARACTER SET ascii2, b VARCHAR(10) CHARACTER SET latin1);
INSERT INTO t1 VALUES ('a','a'),('b','b'); INSERT INTO t1 VALUES ('a','a'),('b','b');
# should not give "illegal collation" error SELECT * FROM t1 WHERE a=b;
ALTER TABLE t1 MODIFY a VARCHAR(10) CHARACTER SET ascii2 COLLATE ascii2_bin2;
SELECT * FROM t1 WHERE a=b; SELECT * FROM t1 WHERE a=b;
ALTER TABLE t1 MODIFY a VARCHAR(10) CHARACTER SET ascii2 COLLATE ascii2_bin; ALTER TABLE t1 MODIFY a VARCHAR(10) CHARACTER SET ascii2 COLLATE ascii2_bin;
# should not give "illegal collation" error SELECT * FROM t1 WHERE a=b;
ALTER TABLE t1 MODIFY a VARCHAR(10) CHARACTER SET ascii2 COLLATE ascii2_general_inherited_ci;
SELECT * FROM t1 WHERE a=b;
ALTER TABLE t1 MODIFY a VARCHAR(10) CHARACTER SET ascii2 COLLATE ascii2_general_inherited2_ci;
SELECT * FROM t1 WHERE a=b;
DROP TABLE t1;
--echo #
--echo # Testing that in case of two binary collations
--echo # "BINARY" in a column definition uses the collation with the least id
--echo #
CREATE TABLE t1 (a VARCHAR(10) CHARACTER SET ascii2 BINARY);
INSERT INTO t1 VALUES ('test');
SELECT COLLATION(a) FROM t1;
DROP TABLE t1;
--echo #
--echo # Testing mixing of two binary collations of the same character set
--echo #
CREATE TABLE t1 (
a VARCHAR(10) CHARACTER SET ascii2 COLLATE ascii2_bin,
b VARCHAR(10) CHARACTER SET ascii2 COLLATE ascii2_bin2
);
INSERT INTO t1 VALUES ('a','a');
--error ER_CANT_AGGREGATE_2COLLATIONS
SELECT * FROM t1 WHERE a=b; SELECT * FROM t1 WHERE a=b;
DROP TABLE t1; DROP TABLE t1;
--echo #
--echo # Testing bad collation inheritance
--echo #
--error ER_UNKNOWN_COLLATION
CREATE TABLE t1 (a VARCHAR(10) CHARACTER SET ascii2 COLLATE ascii2_badly_inherited_ci);
--echo # --echo #
--echo # Testing that the MY_CS_CSSORT flag is set properly --echo # Testing that the MY_CS_CSSORT flag is set properly
......
...@@ -197,13 +197,55 @@ static int cs_copy_data(struct charset_info_st *to, CHARSET_INFO *from) ...@@ -197,13 +197,55 @@ static int cs_copy_data(struct charset_info_st *to, CHARSET_INFO *from)
} }
static my_bool simple_8bit_charset_data_is_full(CHARSET_INFO *cs)
{
return cs->ctype && cs->to_upper && cs->to_lower && cs->tab_to_uni;
}
/**
Inherit missing 8bit charset data from another collation.
Arrays pointed by refcs must be in the permanent memory already,
e.g. static memory, or allocated by my_once_xxx().
*/
static void
inherit_charset_data(struct charset_info_st *cs, CHARSET_INFO *refcs)
{
if (!cs->to_upper)
cs->to_upper= refcs->to_upper;
if (!cs->to_lower)
cs->to_lower= refcs->to_lower;
if (!cs->ctype)
cs->ctype= refcs->ctype;
if (!cs->tab_to_uni)
cs->tab_to_uni= refcs->tab_to_uni;
}
static my_bool simple_8bit_collation_data_is_full(CHARSET_INFO *cs)
{
return cs->sort_order || (cs->state & MY_CS_BINSORT);
}
/**
Inherit 8bit simple collation data from another collation.
refcs->sort_order must be in the permanent memory already,
e.g. static memory, or allocated by my_once_xxx().
*/
static void
inherit_collation_data(struct charset_info_st *cs, CHARSET_INFO *refcs)
{
if (!simple_8bit_collation_data_is_full(cs))
cs->sort_order= refcs->sort_order;
}
static my_bool simple_cs_is_full(CHARSET_INFO *cs) static my_bool simple_cs_is_full(CHARSET_INFO *cs)
{ {
return ((cs->csname && cs->tab_to_uni && cs->ctype && cs->to_upper && return cs->number && cs->csname && cs->name &&
cs->to_lower) && simple_8bit_charset_data_is_full(cs) &&
(cs->number && cs->name && (simple_8bit_collation_data_is_full(cs) || cs->tailoring);
(cs->sort_order || (cs->state & MY_CS_BINSORT) )));
} }
...@@ -336,7 +378,7 @@ static int add_collation(struct charset_info_st *cs) ...@@ -336,7 +378,7 @@ static int add_collation(struct charset_info_st *cs)
cs->name= NULL; cs->name= NULL;
cs->state= 0; cs->state= 0;
cs->sort_order= NULL; cs->sort_order= NULL;
cs->state= 0; cs->tailoring= NULL;
} }
return MY_XML_OK; return MY_XML_OK;
} }
...@@ -631,6 +673,39 @@ const char *get_charset_name(uint charset_number) ...@@ -631,6 +673,39 @@ const char *get_charset_name(uint charset_number)
} }
static CHARSET_INFO *inheritance_source_by_id(CHARSET_INFO *cs, uint refid)
{
CHARSET_INFO *refcs;
return refid && refid != cs->number &&
(refcs= all_charsets[refid]) &&
(refcs->state & MY_CS_AVAILABLE) ? refcs : NULL;
}
static CHARSET_INFO *find_collation_data_inheritance_source(CHARSET_INFO *cs)
{
const char *beg, *end;
if (cs->tailoring &&
!strncmp(cs->tailoring, "[import ", 8) &&
(end= strchr(cs->tailoring + 8, ']')) &&
(beg= cs->tailoring + 8) + MY_CS_NAME_SIZE > end)
{
char name[MY_CS_NAME_SIZE + 1];
memcpy(name, beg, end - beg);
name[end - beg]= '\0';
return inheritance_source_by_id(cs, get_collation_number(name));
}
return NULL;
}
static CHARSET_INFO *find_charset_data_inheritance_source(CHARSET_INFO *cs)
{
uint refid= get_charset_number_internal(cs->csname, MY_CS_PRIMARY);
return inheritance_source_by_id(cs, refid);
}
static CHARSET_INFO * static CHARSET_INFO *
get_internal_charset(MY_CHARSET_LOADER *loader, uint cs_number, myf flags) get_internal_charset(MY_CHARSET_LOADER *loader, uint cs_number, myf flags)
{ {
...@@ -665,6 +740,19 @@ get_internal_charset(MY_CHARSET_LOADER *loader, uint cs_number, myf flags) ...@@ -665,6 +740,19 @@ get_internal_charset(MY_CHARSET_LOADER *loader, uint cs_number, myf flags)
{ {
if (!(cs->state & MY_CS_READY)) if (!(cs->state & MY_CS_READY))
{ {
if (!simple_8bit_charset_data_is_full(cs))
{
CHARSET_INFO *refcs= find_charset_data_inheritance_source(cs);
if (refcs)
inherit_charset_data(cs, refcs);
}
if (!simple_8bit_collation_data_is_full(cs))
{
CHARSET_INFO *refcl= find_collation_data_inheritance_source(cs);
if (refcl)
inherit_collation_data(cs, refcl);
}
if ((cs->cset->init && cs->cset->init(cs, loader)) || if ((cs->cset->init && cs->cset->init(cs, loader)) ||
(cs->coll->init && cs->coll->init(cs, loader))) (cs->coll->init && cs->coll->init(cs, loader)))
{ {
......
...@@ -2078,6 +2078,9 @@ bool DTCollation::aggregate(const DTCollation &dt, uint flags) ...@@ -2078,6 +2078,9 @@ bool DTCollation::aggregate(const DTCollation &dt, uint flags)
set(0, DERIVATION_NONE, 0); set(0, DERIVATION_NONE, 0);
return 1; return 1;
} }
if (collation->state & MY_CS_BINSORT &&
dt.collation->state & MY_CS_BINSORT)
return 1;
if (collation->state & MY_CS_BINSORT) if (collation->state & MY_CS_BINSORT)
return 0; return 0;
if (dt.collation->state & MY_CS_BINSORT) if (dt.collation->state & MY_CS_BINSORT)
......
...@@ -119,7 +119,10 @@ static void simple_cs_copy_data(struct charset_info_st *to, CHARSET_INFO *from) ...@@ -119,7 +119,10 @@ static void simple_cs_copy_data(struct charset_info_st *to, CHARSET_INFO *from)
if (from->name) if (from->name)
to->name= strdup(from->name); to->name= strdup(from->name);
if (from->tailoring)
to->tailoring= strdup(from->tailoring);
if (from->ctype) if (from->ctype)
to->ctype= (uchar*) mdup((char*) from->ctype, MY_CS_CTYPE_TABLE_SIZE); to->ctype= (uchar*) mdup((char*) from->ctype, MY_CS_CTYPE_TABLE_SIZE);
if (from->to_lower) if (from->to_lower)
...@@ -144,30 +147,60 @@ static void simple_cs_copy_data(struct charset_info_st *to, CHARSET_INFO *from) ...@@ -144,30 +147,60 @@ static void simple_cs_copy_data(struct charset_info_st *to, CHARSET_INFO *from)
} }
static void inherit_data(struct charset_info_st *cs, CHARSET_INFO *refcs) /*
cs->xxx arrays can be NULL in case when a collation has an entry only
in Index.xml and has no entry in csname.xml (e.g. in case of a binary
collation or a collation using <import> command).
refcs->xxx arrays can be NULL if <import> refers to a collation
which is not defined in csname.xml, e.g. an always compiled collation
such as latin1_swedish_ci.
*/
static void inherit_charset_data(struct charset_info_st *cs,
CHARSET_INFO *refcs)
{ {
if (refcs->ctype && cs->state|= (refcs->state & (MY_CS_PUREASCII|MY_CS_NONASCII));
if (refcs->ctype && cs->ctype &&
!memcmp(cs->ctype, refcs->ctype, MY_CS_CTYPE_TABLE_SIZE)) !memcmp(cs->ctype, refcs->ctype, MY_CS_CTYPE_TABLE_SIZE))
cs->ctype= NULL; cs->ctype= NULL;
if (refcs->to_lower && if (refcs->to_lower && cs->to_lower &&
!memcmp(cs->to_lower, refcs->to_lower, MY_CS_TO_LOWER_TABLE_SIZE)) !memcmp(cs->to_lower, refcs->to_lower, MY_CS_TO_LOWER_TABLE_SIZE))
cs->to_lower= NULL; cs->to_lower= NULL;
if (refcs->to_upper && if (refcs->to_upper && cs->to_upper &&
!memcmp(cs->to_upper, refcs->to_upper, MY_CS_TO_LOWER_TABLE_SIZE)) !memcmp(cs->to_upper, refcs->to_upper, MY_CS_TO_LOWER_TABLE_SIZE))
cs->to_upper= NULL; cs->to_upper= NULL;
if (refcs->tab_to_uni && if (refcs->tab_to_uni && cs->tab_to_uni &&
!memcmp(cs->tab_to_uni, refcs->tab_to_uni, !memcmp(cs->tab_to_uni, refcs->tab_to_uni,
MY_CS_TO_UNI_TABLE_SIZE * sizeof(uint16))) MY_CS_TO_UNI_TABLE_SIZE * sizeof(uint16)))
cs->tab_to_uni= NULL; cs->tab_to_uni= NULL;
} }
static CHARSET_INFO *find_charset_data_inheritance_source(CHARSET_INFO *cs)
{
CHARSET_INFO *refcs;
uint refid= get_charset_number_internal(cs->csname, MY_CS_PRIMARY);
return refid && refid != cs->number &&
(refcs= &all_charsets[refid]) &&
(refcs->state & MY_CS_LOADED) ? refcs : NULL;
}
/**
Detect if "cs" needs further loading from csname.xml
@param cs - the character set pointer
@retval FALSE - if the current data (e.g. loaded from from Index.xml)
is not enough to dump the character set and requires
further reading from the csname.xml file.
@retval TRUE - if the current data is enough to dump,
no reading of csname.xml is needed.
*/
static my_bool simple_cs_is_full(CHARSET_INFO *cs) static my_bool simple_cs_is_full(CHARSET_INFO *cs)
{ {
return ((cs->csname && cs->tab_to_uni && cs->ctype && cs->to_upper && return ((cs->csname && cs->tab_to_uni && cs->ctype && cs->to_upper &&
cs->to_lower) && cs->to_lower) &&
(cs->number && cs->name && (cs->number && cs->name &&
(cs->sort_order || (cs->state & MY_CS_BINSORT)))); (cs->sort_order || cs->tailoring || (cs->state & MY_CS_BINSORT))));
} }
static int add_collation(struct charset_info_st *cs) static int add_collation(struct charset_info_st *cs)
...@@ -183,6 +216,7 @@ static int add_collation(struct charset_info_st *cs) ...@@ -183,6 +216,7 @@ static int add_collation(struct charset_info_st *cs)
cs->number= 0; cs->number= 0;
cs->name= NULL; cs->name= NULL;
cs->tailoring= NULL;
cs->state= 0; cs->state= 0;
cs->sort_order= NULL; cs->sort_order= NULL;
cs->state= 0; cs->state= 0;
...@@ -255,6 +289,55 @@ void print_arrays(FILE *f, CHARSET_INFO *cs) ...@@ -255,6 +289,55 @@ void print_arrays(FILE *f, CHARSET_INFO *cs)
} }
/**
Print an array member of a CHARSET_INFO.
@param f - the file to print into
@param cs0 - reference to the CHARSET_INFO to print
@param array0 - pointer to the array data (can be NULL)
@param cs1 - reference to the CHARSET_INFO that the data
can be inherited from (e.g. primary collation)
@param array1 - pointer to the array data in cs1 (can be NULL)
@param name - name of the member
If array0 is not null, then the CHARSET_INFO being dumped has its
own array (e.g. the default collation for the character set).
We print the name of this array using cs0->name and return.
If array1 is not null, then the CHARSET_INFO being dumpled reuses
the array from another collation. We print the name of the array of
the referenced collation using cs1->name and return.
Otherwise (if both array0 and array1 are NULL), we have a collation
of a character set whose primary collation is not available now,
and which does not have its own entry in csname.xml file.
For example, Index.xml has this entry:
<collation name="latin1_swedish_ci_copy">
<rules>
<import source="latin1_swedish_ci"/>
</rules>
</collation>
and latin1.xml does not have entries for latin1_swedish_ci_copy.
In such cases we print NULL as a pointer to the array.
It will be set to a not-null data during the first initialization
by the inherit_charset_data() call (see mysys/charset.c for details).
*/
static void
print_array_ref(FILE *f,
CHARSET_INFO *cs0, const void *array0,
CHARSET_INFO *cs1, const void *array1,
const char *name)
{
CHARSET_INFO *cs= array0 ? cs0 : array1 ? cs1 : NULL;
if (cs)
fprintf(f," %s_%s, /* %s */\n",
name, cs->name, name);
else
fprintf(f," NULL, /* %s */\n", name);
}
void dispcset(FILE *f,CHARSET_INFO *cs) void dispcset(FILE *f,CHARSET_INFO *cs)
{ {
fprintf(f,"{\n"); fprintf(f,"{\n");
...@@ -272,21 +355,23 @@ void dispcset(FILE *f,CHARSET_INFO *cs) ...@@ -272,21 +355,23 @@ void dispcset(FILE *f,CHARSET_INFO *cs)
fprintf(f," \"%s\", /* cset name */\n",cs->csname); fprintf(f," \"%s\", /* cset name */\n",cs->csname);
fprintf(f," \"%s\", /* coll name */\n",cs->name); fprintf(f," \"%s\", /* coll name */\n",cs->name);
fprintf(f," \"\", /* comment */\n"); fprintf(f," \"\", /* comment */\n");
fprintf(f," NULL, /* tailoring */\n"); if (cs->tailoring)
fprintf(f, " \"%s\", /* tailoring */\n", cs->tailoring);
fprintf(f," ctype_%s, /* ctype */\n", else
cs->ctype ? cs->name : srccs->name); fprintf(f," NULL, /* tailoring */\n");
fprintf(f," to_lower_%s, /* lower */\n",
cs->to_lower ? cs->name : srccs->name); print_array_ref(f, cs, cs->ctype, srccs, srccs->ctype, "ctype");
fprintf(f," to_upper_%s, /* upper */\n", print_array_ref(f, cs, cs->to_lower, srccs, srccs->to_lower, "to_lower");
cs->to_upper ? cs->name : srccs->name); print_array_ref(f, cs, cs->to_upper, srccs, srccs->to_upper, "to_upper");
if (cs->sort_order) if (cs->sort_order)
fprintf(f," sort_order_%s, /* sort_order */\n",cs->name); fprintf(f," sort_order_%s, /* sort_order */\n",cs->name);
else else
fprintf(f," NULL, /* sort_order */\n"); fprintf(f," NULL, /* sort_order */\n");
fprintf(f," NULL, /* uca */\n"); fprintf(f," NULL, /* uca */\n");
fprintf(f," to_uni_%s, /* to_uni */\n",
cs->tab_to_uni ? cs->name : srccs->name); print_array_ref(f, cs, cs->tab_to_uni, srccs, srccs->tab_to_uni, "to_uni");
} }
else else
{ {
...@@ -403,14 +488,13 @@ main(int argc, char **argv __attribute__((unused))) ...@@ -403,14 +488,13 @@ main(int argc, char **argv __attribute__((unused)))
{ {
if (cs->state & MY_CS_LOADED) if (cs->state & MY_CS_LOADED)
{ {
uint refid= get_charset_number_internal(cs->csname, MY_CS_PRIMARY); CHARSET_INFO *refcs= find_charset_data_inheritance_source(cs);
cs->state|= my_8bit_charset_flags_from_data(cs) | cs->state|= my_8bit_charset_flags_from_data(cs) |
my_8bit_collation_flags_from_data(cs); my_8bit_collation_flags_from_data(cs);
if (refid && cs->number != refid) if (refcs)
{ {
CHARSET_INFO *refcs= &all_charsets[refid]; refids[cs->number]= refcs->number;
refids[cs->number]= refid; inherit_charset_data(cs, refcs);
inherit_data(cs, refcs);
} }
fprintf(f,"#ifdef HAVE_CHARSET_%s\n",cs->csname); fprintf(f,"#ifdef HAVE_CHARSET_%s\n",cs->csname);
print_arrays(f, cs); print_arrays(f, cs);
......
...@@ -1417,6 +1417,8 @@ my_cset_init_8bit(struct charset_info_st *cs, MY_CHARSET_LOADER *loader) ...@@ -1417,6 +1417,8 @@ my_cset_init_8bit(struct charset_info_st *cs, MY_CHARSET_LOADER *loader)
cs->caseup_multiply= 1; cs->caseup_multiply= 1;
cs->casedn_multiply= 1; cs->casedn_multiply= 1;
cs->pad_char= ' '; cs->pad_char= ' ';
if (!cs->to_lower || !cs->to_upper || !cs->ctype || !cs->tab_to_uni)
return TRUE;
return create_fromuni(cs, loader); return create_fromuni(cs, loader);
} }
...@@ -1442,6 +1444,8 @@ static void set_max_sort_char(struct charset_info_st *cs) ...@@ -1442,6 +1444,8 @@ static void set_max_sort_char(struct charset_info_st *cs)
static my_bool my_coll_init_simple(struct charset_info_st *cs, static my_bool my_coll_init_simple(struct charset_info_st *cs,
MY_CHARSET_LOADER *loader __attribute__((unused))) MY_CHARSET_LOADER *loader __attribute__((unused)))
{ {
if (!cs->sort_order)
return TRUE;
cs->state|= my_8bit_collation_flags_from_data(cs); cs->state|= my_8bit_collation_flags_from_data(cs);
set_max_sort_char(cs); set_max_sort_char(cs);
return FALSE; return FALSE;
......
...@@ -88,6 +88,8 @@ struct my_cs_file_section_st ...@@ -88,6 +88,8 @@ struct my_cs_file_section_st
#define _CS_CL_SUPPRESS_CONTRACTIONS 101 #define _CS_CL_SUPPRESS_CONTRACTIONS 101
#define _CS_CL_OPTIMIZE 102 #define _CS_CL_OPTIMIZE 102
#define _CS_CL_SHIFT_AFTER_METHOD 103 #define _CS_CL_SHIFT_AFTER_METHOD 103
#define _CS_CL_RULES_IMPORT 104
#define _CS_CL_RULES_IMPORT_SOURCE 105
/* Collation Settings */ /* Collation Settings */
...@@ -188,6 +190,8 @@ static const struct my_cs_file_section_st sec[] = ...@@ -188,6 +190,8 @@ static const struct my_cs_file_section_st sec[] =
{_CS_CL_SUPPRESS_CONTRACTIONS, "charsets/charset/collation/suppress_contractions"}, {_CS_CL_SUPPRESS_CONTRACTIONS, "charsets/charset/collation/suppress_contractions"},
{_CS_CL_OPTIMIZE, "charsets/charset/collation/optimize"}, {_CS_CL_OPTIMIZE, "charsets/charset/collation/optimize"},
{_CS_CL_SHIFT_AFTER_METHOD, "charsets/charset/collation/shift-after-method"}, {_CS_CL_SHIFT_AFTER_METHOD, "charsets/charset/collation/shift-after-method"},
{_CS_CL_RULES_IMPORT, "charsets/charset/collation/rules/import"},
{_CS_CL_RULES_IMPORT_SOURCE, "charsets/charset/collation/rules/import/source"},
/* Collation Settings */ /* Collation Settings */
{_CS_ST_SETTINGS, "charsets/charset/collation/settings"}, {_CS_ST_SETTINGS, "charsets/charset/collation/settings"},
...@@ -641,6 +645,10 @@ static int cs_value(MY_XML_PARSER *st,const char *attr, size_t len) ...@@ -641,6 +645,10 @@ static int cs_value(MY_XML_PARSER *st,const char *attr, size_t len)
rc= tailoring_append(st, "[version %.*s]", len, attr); rc= tailoring_append(st, "[version %.*s]", len, attr);
break; break;
case _CS_CL_RULES_IMPORT_SOURCE:
rc= tailoring_append(st, "[import %.*s]", len, attr);
break;
case _CS_CL_SUPPRESS_CONTRACTIONS: case _CS_CL_SUPPRESS_CONTRACTIONS:
rc= tailoring_append(st, "[suppress contractions %.*s]", len, attr); rc= tailoring_append(st, "[suppress contractions %.*s]", len, attr);
break; break;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment