/* Copyright (C) 2004 MySQL AB 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 the Free Software Foundation; either version 2 of the License, or (at your option) any later version. 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 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ /* Written by Anjuta Widenius */ /* Creates one include file and multiple language-error message files from one multi-language text file. */ #include #include #include #include #include #include #include #define MAX_ROWS 1000 #define HEADER_LENGTH 32 /* Length of header in errmsg.sys */ #define DEFAULT_CHARSET_DIR "../sql/share/charsets" #define ER_PREFIX "ER_" static char *OUTFILE= (char*) "errmsg.sys"; static char *HEADERFILE= (char*) "mysqld_error.h"; static char *STATEFILE= (char*) "sql_state.h"; static char *TXTFILE= (char*) "../sql/share/errmsg.txt"; static char *DATADIR= (char*) "../sql/share/"; static char *default_dbug_option= (char*) "d:t:O,/tmp/comp_err.trace"; /* Header for errmsg.sys files */ uchar file_head[]= { 254, 254, 2, 1 }; /* Store positions to each error message row to store in errmsg.sys header */ uint file_pos[MAX_ROWS]; const char *empty_string= ""; /* For empty states */ /* Default values for command line options. See getopt structure for defintions for these. */ const char *default_language= "eng"; int er_offset= 1000; my_bool info_flag= 0; /* Storage of one error message row (for one language) */ struct message { char *lang_short_name; char *text; }; /* Storage for languages and charsets (from start of error text file) */ struct languages { char *lang_long_name; /* full name of the language */ char *lang_short_name; /* abbreviation of the lang. */ char *charset; /* Character set name */ struct languages *next_lang; /* Pointer to next language */ }; /* Name, code and texts (for all lang) for one error message */ struct errors { const char *er_name; /* Name of the error (ER_HASHCK) */ int d_code; /* Error code number */ const char *sql_code1; /* sql state */ const char *sql_code2; /* ODBC state */ struct errors *next_error; /* Pointer to next error */ DYNAMIC_ARRAY msg; /* All language texts for this error */ }; static struct my_option my_long_options[]= { #ifdef DBUG_OFF {"debug", '#', "This is a non-debug version. Catch this and exit", 0, 0, 0, GET_DISABLED, OPT_ARG, 0, 0, 0, 0, 0, 0}, #else {"debug", '#', "Output debug log", (gptr *) & default_dbug_option, (gptr *) & default_dbug_option, 0, GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0}, #endif {"debug-info", 'T', "Print some debug info at exit.", (gptr *) & info_flag, (gptr *) & info_flag, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0}, {"help", '?', "Displays this help and exits.", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"version", 'V', "Prints version", 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}, {"charset", 'C', "Charset dir", (gptr *) & charsets_dir, (gptr *) & charsets_dir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"in_file", 'F', "Input file", (gptr *) & TXTFILE, (gptr *) & TXTFILE, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"out_dir", 'D', "Output base directory", (gptr *) & DATADIR, (gptr *) & DATADIR, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"out_file", 'O', "Output filename (errmsg.sys)", (gptr *) & OUTFILE, (gptr *) & OUTFILE, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"header_file", 'H', "mysqld_error.h file ", (gptr *) & HEADERFILE, (gptr *) & HEADERFILE, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {"state_file", 'S', "sql_state.h file", (gptr *) & STATEFILE, (gptr *) & STATEFILE, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0} }; static struct languages *parse_charset_string(char *str); static struct errors *parse_error_string(char *ptr, int er_count); static struct message *parse_message_string(struct message *new_message, char *str); static struct message *find_message(struct errors *err, const char *lang); static int parse_input_file(const char *file_name, struct errors **top_error, struct languages **top_language); static int get_options(int *argc, char ***argv); static void print_version(void); static void usage(void); static my_bool get_one_option(int optid, const struct my_option *opt, char *argument); static char *parse_text_line(char *pos); static int copy_rows(FILE * to, char *row, int row_nr, long start_pos); static char *parse_default_language(char *str); static uint parse_error_offset(char *str); static char *skip_delimiters(char *str); static char *get_word(char **str); static char *find_end_of_word(char *str); static void clean_up(struct languages *lang_head, struct errors *error_head); static int create_header_files(struct errors *error_head); static int create_sys_files(struct languages *lang_head, struct errors *error_head, uint row_count); int main(int argc, char *argv[]) { MY_INIT(argv[0]); { uint row_count; struct errors *error_head; struct languages *lang_head; DBUG_ENTER("main"); charsets_dir= DEFAULT_CHARSET_DIR; if (get_options(&argc, &argv)) DBUG_RETURN(1); if (!(row_count= parse_input_file(TXTFILE, &error_head, &lang_head))) { fprintf(stderr, "Failed to parse input file %s\n", TXTFILE); DBUG_RETURN(1); } if (lang_head == NULL || error_head == NULL) { fprintf(stderr, "Failed to parse input file %s\n", TXTFILE); DBUG_RETURN(1); } if (create_header_files(error_head)) { fprintf(stderr, "Failed to create header files\n"); DBUG_RETURN(1); } if (create_sys_files(lang_head, error_head, row_count)) { fprintf(stderr, "Failed to create sys files\n"); DBUG_RETURN(1); } clean_up(lang_head, error_head); my_end(info_flag ? MY_CHECK_ERROR | MY_GIVE_INFO : 0); DBUG_RETURN(0); } } static int create_header_files(struct errors *error_head) { uint er_count= 0; FILE *er_definef, *sql_statef; struct errors *tmp_error; DBUG_ENTER("create_header_files"); if (!(er_definef= my_fopen(HEADERFILE, O_WRONLY, MYF(MY_WME)))) { DBUG_RETURN(1); } if (!(sql_statef= my_fopen(STATEFILE, O_WRONLY, MYF(MY_WME)))) { my_fclose(er_definef, MYF(0)); DBUG_RETURN(1); } fprintf(er_definef, "/* Autogenerated file, please don't edit */\n\n"); fprintf(sql_statef, "/* Autogenerated file, please don't edit */\n\n"); for (tmp_error= error_head; tmp_error; tmp_error= tmp_error->next_error) { /* generating mysqld_error.h fprintf() will automaticly add \r on windows */ fprintf(er_definef, "#define %s %d\n", tmp_error->er_name, tmp_error->d_code); /* generating sql_state.h file */ if (tmp_error->sql_code1[0] || tmp_error->sql_code2[0]) fprintf(sql_statef, "%-40s,\"%s\", \"%s\",\n", tmp_error->er_name, tmp_error->sql_code1, tmp_error->sql_code2); er_count++; } /* finishing off with mysqld_error.h */ fprintf(er_definef, "#define ER_ERROR_MESSAGES %d\n", er_count); my_fclose(er_definef, MYF(0)); my_fclose(sql_statef, MYF(0)); DBUG_RETURN(0); } static int create_sys_files(struct languages *lang_head, struct errors *error_head, uint row_count) { FILE *to; uint csnum= 0, length, i, row_nr; uchar head[32]; char outfile[FN_REFLEN], *outfile_end; long start_pos; struct message *tmp; struct languages *tmp_lang; struct errors *tmp_error; MY_STAT stat_info; DBUG_ENTER("create_sys_files"); /* going over all languages and assembling corresponding error messages */ for (tmp_lang= lang_head; tmp_lang; tmp_lang= tmp_lang->next_lang) { /* setting charset name */ if (!(csnum= get_charset_number(tmp_lang->charset, MY_CS_PRIMARY))) { fprintf(stderr, "Unknown charset '%s' in '%s'\n", tmp_lang->charset, TXTFILE); DBUG_RETURN(1); } outfile_end= strxmov(outfile, DATADIR, tmp_lang->lang_long_name, NullS); if (!my_stat(outfile, &stat_info,MYF(0))) { if (my_mkdir(outfile, 0777,MYF(0)) < 0) { fprintf(stderr, "Can't create output directory for %s\n", outfile); DBUG_RETURN(1); } } strxmov(outfile_end, FN_ROOTDIR, OUTFILE, NullS); if (!(to= my_fopen(outfile, O_WRONLY | FILE_BINARY, MYF(MY_WME)))) DBUG_RETURN(1); /* 2 is for 2 bytes to store row position / error message */ start_pos= (long) (HEADER_LENGTH + row_count * 2); fseek(to, start_pos, 0); row_nr= 0; for (tmp_error= error_head; tmp_error; tmp_error= tmp_error->next_error) { /* dealing with messages */ tmp= find_message(tmp_error, tmp_lang->lang_short_name); if (!tmp) { fprintf(stderr, "Did not find message for %s neither in %s nor in default " "language\n", tmp_error->er_name, tmp_lang->lang_short_name); goto err; } if (copy_rows(to, tmp->text, row_nr, start_pos)) { fprintf(stderr, "Failed to copy rows to %s\n", outfile); goto err; } row_nr++; } /* continue with header of the errmsg.sys file */ length= ftell(to) - HEADER_LENGTH - row_count * 2; bzero((gptr) head, HEADER_LENGTH); bmove((byte *) head, (byte *) file_head, 4); head[4]= 1; int2store(head + 6, length); int2store(head + 8, row_count); head[30]= csnum; my_fseek(to, 0l, MY_SEEK_SET, MYF(0)); if (my_fwrite(to, head, HEADER_LENGTH, MYF(MY_WME | MY_FNABP))) goto err; for (i= 0; i < row_count; i++) { int2store(head, file_pos[i]); if (my_fwrite(to, head, 2, MYF(MY_WME | MY_FNABP))) goto err; } my_fclose(to, MYF(0)); } DBUG_RETURN(0); err: my_fclose(to, MYF(0)); DBUG_RETURN(1); } static void clean_up(struct languages *lang_head, struct errors *error_head) { struct languages *tmp_lang, *next_language; struct errors *tmp_error, *next_error; uint count, i; my_free((gptr) default_language, MYF(0)); for (tmp_lang= lang_head; tmp_lang; tmp_lang= next_language) { next_language= tmp_lang->next_lang; my_free(tmp_lang->lang_short_name, MYF(0)); my_free(tmp_lang->lang_long_name, MYF(0)); my_free(tmp_lang->charset, MYF(0)); my_free((gptr) tmp_lang, MYF(0)); } for (tmp_error= error_head; tmp_error; tmp_error= next_error) { next_error= tmp_error->next_error; count= (tmp_error->msg).elements; for (i= 0; i < count; i++) { struct message *tmp; tmp= dynamic_element(&tmp_error->msg, i, struct message*); my_free((gptr) tmp->lang_short_name, MYF(0)); my_free((gptr) tmp->text, MYF(0)); } delete_dynamic(&tmp_error->msg); if (tmp_error->sql_code1[0]) my_free((gptr) tmp_error->sql_code1, MYF(0)); if (tmp_error->sql_code2[0]) my_free((gptr) tmp_error->sql_code2, MYF(0)); my_free((gptr) tmp_error->er_name, MYF(0)); my_free((gptr) tmp_error, MYF(0)); } } static int parse_input_file(const char *file_name, struct errors **top_error, struct languages **top_lang) { FILE *file; char *str, buff[1000]; struct errors *current_error= 0, **tail_error= top_error; struct message current_message; int rcount= 0; DBUG_ENTER("parse_input_file"); if (!(file= my_fopen(file_name, O_RDONLY | O_SHARE, MYF(MY_WME)))) DBUG_RETURN(0); while ((str= fgets(buff, sizeof(buff), file))) { if (is_prefix(str, "language")) { if (!(*top_lang= parse_charset_string(str))) { fprintf(stderr, "Failed to parse the charset string!\n"); DBUG_RETURN(0); } continue; } if (is_prefix(str, "start-error-number")) { if (!(er_offset= parse_error_offset(str))) { fprintf(stderr, "Failed to parse the error offset string!\n"); DBUG_RETURN(0); } continue; } if (is_prefix(str, "default-language")) { if (!(default_language= parse_default_language(str))) { DBUG_PRINT("info", ("default_slang: %s", default_language)); fprintf(stderr, "Failed to parse the default language line. Aborting\n"); DBUG_RETURN(0); } continue; } if (*str == '\t' || *str == ' ') { /* New error message in another language for previous error */ if (!current_error) { fprintf(stderr, "Error in the input file format\n"); DBUG_RETURN(0); } if (!parse_message_string(¤t_message, str)) { fprintf(stderr, "Failed to parse message string for error '%s'", current_error->er_name); DBUG_RETURN(0); } if (insert_dynamic(¤t_error->msg, (byte *) & current_message)) DBUG_RETURN(0); continue; } if (is_prefix(str, ER_PREFIX)) { if (!(current_error= parse_error_string(str, rcount))) { fprintf(stderr, "Failed to parse the error name string\n"); DBUG_RETURN(0); } rcount++; /* Count number of unique errors */ /* add error to the list */ *tail_error= current_error; tail_error= ¤t_error->next_error; continue; } if (*str == '#' || *str == '\n') continue; /* skip comment or empty lines */ fprintf(stderr, "Wrong input file format. Stop!\nLine: %s\n", str); DBUG_RETURN(0); } *tail_error= 0; /* Mark end of list */ my_fclose(file, MYF(0)); DBUG_RETURN(rcount); } static uint parse_error_offset(char *str) { char *soffset, *end; int error; uint ioffset; DBUG_ENTER("parse_error_offset"); /* skipping the "start-error-number" keyword and spaces after it */ str= find_end_of_word(str); str= skip_delimiters(str); if (!*str) DBUG_RETURN(0); /* Unexpected EOL: No error number after the keyword */ /* reading the error offset */ if (!(soffset= get_word(&str))) DBUG_RETURN(0); /* OOM: Fatal error */ DBUG_PRINT("info", ("default_error_offset: %s", soffset)); /* skipping space(s) and/or tabs after the error offset */ str= skip_delimiters(str); DBUG_PRINT("info", ("str: %s", str)); if (*str) { /* The line does not end with the error offset -> error! */ fprintf(stderr, "The error offset line does not end with an error offset"); DBUG_RETURN(0); } DBUG_PRINT("info", ("str: %s", str)); end= 0; ioffset= (uint) my_strtoll10(soffset, &end, &error); my_free((gptr) soffset, MYF(0)); DBUG_RETURN(ioffset); } /* Parsing of the default language line. e.g. "default-lanuage eng" */ static char *parse_default_language(char *str) { char *slang; DBUG_ENTER("parse_default_language"); /* skipping the "default_language" keyword */ str= find_end_of_word(str); /* skipping space(s) and/or tabs after the keyword */ str= skip_delimiters(str); if (!*str) { fprintf(stderr, "Unexpected EOL: No short language name after the keyword\n"); DBUG_RETURN(0); } /* reading the short language tag */ if (!(slang= get_word(&str))) DBUG_RETURN(0); /* OOM: Fatal error */ DBUG_PRINT("info", ("default_slang: %s", slang)); str= skip_delimiters(str); DBUG_PRINT("info", ("str: %s", str)); if (*str) { fprintf(stderr, "The default language line does not end with short language " "name\n"); DBUG_RETURN(0); } DBUG_PRINT("info", ("str: %s", str)); DBUG_RETURN(slang); } /* For given error finds message on given language, if does not exist, returns english. */ static struct message *find_message(struct errors *err, const char *lang) { struct message *tmp, *return_val= 0; uint i, count; DBUG_ENTER("find_message"); count= (err->msg).elements; for (i= 0; i < count; i++) { tmp= dynamic_element(&err->msg, i, struct message*); if (!strcmp(tmp->lang_short_name, lang)) DBUG_RETURN(tmp); if (!strcmp(tmp->lang_short_name, default_language)) { DBUG_ASSERT(tmp->text[0] != 0); return_val= tmp; } } DBUG_RETURN(return_val); } /* Skips spaces and or tabs till the beginning of the next word Returns pointer to the beginning of the first character of the word */ static char *skip_delimiters(char *str) { DBUG_ENTER("skip_delimiters"); for (; *str == ' ' || *str == ',' || *str == '\t' || *str == '\r' || *str == '\n' || *str == '='; str++) ; DBUG_RETURN(str); } /* Skips all characters till meets with space, or tab, or EOL */ static char *find_end_of_word(char *str) { DBUG_ENTER("find_end_of_word"); for (; *str != ' ' && *str != '\t' && *str != '\n' && *str != '\r' && *str && *str != ',' && *str != ';' && *str != '='; str++) ; DBUG_RETURN(str); } /* Read the word starting from *str */ static char *get_word(char **str) { char *start= *str; DBUG_ENTER("get_word"); *str= find_end_of_word(start); DBUG_RETURN(my_strdup_with_length(start, (uint) (*str - start), MYF(MY_WME | MY_FAE))); } /* Parsing the string with short_lang - message text. Code - to remember to which error does the text belong */ static struct message *parse_message_string(struct message *new_message, char *str) { char *start; DBUG_ENTER("parse_message_string"); DBUG_PRINT("enter", ("str: %s", str)); /*skip space(s) and/or tabs in the beginning */ while (*str == ' ' || *str == '\t' || *str == '\n') str++; if (!*str) { /* It was not a message line, but an empty line. */ DBUG_PRINT("info", ("str: %s", str)); DBUG_RETURN(0); } /* reading the short lang */ start= str; while (*str != ' ' && *str != '\t' && *str) str++; if (!(new_message->lang_short_name= my_strdup_with_length(start, (uint) (str - start), MYF(MY_WME | MY_FAE)))) DBUG_RETURN(0); /* Fatal error */ DBUG_PRINT("info", ("msg_slang: %s", new_message->lang_short_name)); /*skip space(s) and/or tabs after the lang */ while (*str == ' ' || *str == '\t' || *str == '\n') str++; if (*str != '"') { fprintf(stderr, "Unexpected EOL"); DBUG_PRINT("info", ("str: %s", str)); DBUG_RETURN(0); } /* reading the text */ start= str + 1; str= parse_text_line(start); if (!(new_message->text= my_strdup_with_length(start, (uint) (str - start), MYF(MY_WME | MY_FAE)))) DBUG_RETURN(0); /* Fatal error */ DBUG_PRINT("info", ("msg_text: %s", new_message->text)); DBUG_RETURN(new_message); } /* Parsing the string with error name and codes; returns the pointer to the errors struct */ static struct errors *parse_error_string(char *str, int er_count) { struct errors *new_error; char *start; DBUG_ENTER("parse_error_string"); DBUG_PRINT("enter", ("str: %s", str)); /* create a new a element */ new_error= (struct errors *) my_malloc(sizeof(*new_error), MYF(MY_WME)); if (my_init_dynamic_array(&new_error->msg, sizeof(struct message), 0, 0)) DBUG_RETURN(0); /* OOM: Fatal error */ /* getting the error name */ start= str; str= skip_delimiters(str); if (!(new_error->er_name= get_word(&str))) DBUG_RETURN(0); /* OOM: Fatal error */ DBUG_PRINT("info", ("er_name: %s", new_error->er_name)); str= skip_delimiters(str); /* getting the code1 */ new_error->d_code= er_offset + er_count; DBUG_PRINT("info", ("d_code: %d", new_error->d_code)); str= skip_delimiters(str); /* if we reached EOL => no more codes, but this can happen */ if (!*str) { new_error->sql_code1= empty_string; new_error->sql_code2= empty_string; DBUG_PRINT("info", ("str: %s", str)); DBUG_RETURN(new_error); } /* getting the sql_code 1 */ if (!(new_error->sql_code1= get_word(&str))) DBUG_RETURN(0); /* OOM: Fatal error */ DBUG_PRINT("info", ("sql_code1: %s", new_error->sql_code1)); str= skip_delimiters(str); /* if we reached EOL => no more codes, but this can happen */ if (!*str) { new_error->sql_code2= empty_string; DBUG_PRINT("info", ("str: %s", str)); DBUG_RETURN(new_error); } /* getting the sql_code 2 */ if (!(new_error->sql_code2= get_word(&str))) DBUG_RETURN(0); /* OOM: Fatal error */ DBUG_PRINT("info", ("sql_code2: %s", new_error->sql_code2)); str= skip_delimiters(str); if (*str) { fprintf(stderr, "The error line did not end with sql/odbc code!"); DBUG_RETURN(0); } DBUG_RETURN(new_error); } /* Parsing the string with charset/full lang name/short lang name; returns pointer to the language structure */ static struct languages *parse_charset_string(char *str) { struct languages *head=0, *new_lang; DBUG_ENTER("parse_charset_string"); DBUG_PRINT("enter", ("str: %s", str)); /* skip over keyword */ str= find_end_of_word(str); if (!*str) { /* unexpected EOL */ DBUG_PRINT("info", ("str: %s", str)); DBUG_RETURN(0); } str= skip_delimiters(str); if (!(*str != ';' && *str)) DBUG_RETURN(0); do { /*creating new element of the linked list */ new_lang= (struct languages *) my_malloc(sizeof(*new_lang), MYF(MY_WME)); new_lang->next_lang= head; head= new_lang; /* get the full language name */ if (!(new_lang->lang_long_name= get_word(&str))) DBUG_RETURN(0); /* OOM: Fatal error */ DBUG_PRINT("info", ("long_name: %s", new_lang->lang_long_name)); /* getting the short name for language */ str= skip_delimiters(str); if (!*str) DBUG_RETURN(0); /* Error: No space or tab */ if (!(new_lang->lang_short_name= get_word(&str))) DBUG_RETURN(0); /* OOM: Fatal error */ DBUG_PRINT("info", ("short_name: %s", new_lang->lang_short_name)); /* getting the charset name */ str= skip_delimiters(str); if (!(new_lang->charset= get_word(&str))) DBUG_RETURN(0); /* Fatal error */ DBUG_PRINT("info", ("charset: %s", new_lang->charset)); /* skipping space, tub or "," */ str= skip_delimiters(str); } while (*str != ';' && *str); DBUG_PRINT("info", ("long name: %s", new_lang->lang_long_name)); DBUG_RETURN(head); } /* Read options */ static void print_version(void) { DBUG_ENTER("print_version"); printf("%s (Compile errormessage) Ver %s\n", my_progname, "2.0"); DBUG_VOID_RETURN; } static my_bool get_one_option(int optid, const struct my_option *opt __attribute__ ((unused)), char *argument __attribute__ ((unused))) { DBUG_ENTER("get_one_option"); switch (optid) { case 'V': print_version(); exit(0); break; case '?': usage(); exit(0); break; case '#': DBUG_PUSH(argument ? argument : default_dbug_option); break; } DBUG_RETURN(0); } static void usage(void) { DBUG_ENTER("usage"); print_version(); printf("This software comes with ABSOLUTELY NO WARRANTY. This is free " "software,\nand you are welcome to modify and redistribute it under " "the GPL license\nUsage:\n"); my_print_help(my_long_options); my_print_variables(my_long_options); DBUG_VOID_RETURN; } static int get_options(int *argc, char ***argv) { int ho_error; DBUG_ENTER("get_options"); if ((ho_error= handle_options(argc, argv, my_long_options, get_one_option))) DBUG_RETURN(ho_error); DBUG_RETURN(0); } /* Read rows and remember them until row that start with char Converts row as a C-compiler would convert a textstring */ static char *parse_text_line(char *pos) { int i, nr; char *row; row= pos; DBUG_ENTER("parse_text_line"); while (*pos) { if (*pos == '\\') { switch (*++pos) { case '\\': case '"': VOID(strmov(pos - 1, pos)); break; case 'n': pos[-1]= '\n'; VOID(strmov(pos, pos + 1)); break; default: if (*pos >= '0' && *pos < '8') { nr= 0; for (i= 0; i < 3 && (*pos >= '0' && *pos < '8'); i++) nr= nr * 8 + (*(pos++) - '0'); pos -= i; pos[-1]= nr; VOID(strmov(pos, pos + i)); } else if (*pos) VOID(strmov(pos - 1, pos)); /* Remove '\' */ } } else pos++; } while (pos > row + 1 && *pos != '"') pos--; *pos= 0; DBUG_RETURN(pos); } /* Copy rows from memory to file and remember position */ static int copy_rows(FILE *to, char *row, int row_nr, long start_pos) { DBUG_ENTER("copy_rows"); file_pos[row_nr]= (int) (ftell(to) - start_pos); if (fputs(row, to) == EOF || fputc('\0', to) == EOF) { fprintf(stderr, "Can't write to outputfile\n"); DBUG_RETURN(1); } DBUG_RETURN(0); }