4 * This file is part of OpenTTD.
5 * OpenTTD 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, version 2.
6 * OpenTTD 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.
7 * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <http://www.gnu.org/licenses/>.
10 /** @file strgen.cpp Tool to create computer readable (stand-alone) translation files. */
12 #include "../stdafx.h"
13 #include "../core/endian_func.hpp"
14 #include "../string_func.h"
15 #include "../strings_type.h"
16 #include "../misc/getoptdata.h"
17 #include "../table/control_codes.h"
24 #if (!defined(WIN32) && !defined(WIN64)) || defined(__CYGWIN__)
29 #if defined WIN32 || defined __WATCOMC__
31 #endif /* WIN32 || __WATCOMC__ */
38 #endif /* __MORPHOS__ */
40 #include "../table/strgen_tables.h"
42 #include "../safeguards.h"
46 # define LINE_NUM_FMT(s) "%s (%d): warning: %s (" s ")\n"
48 # define LINE_NUM_FMT(s) "%s:%d: " s ": %s\n"
51 void CDECL
strgen_warning(const char *s
, ...)
56 vseprintf(buf
, lastof(buf
), s
, va
);
58 fprintf(stderr
, LINE_NUM_FMT("warning"), _file
, _cur_line
, buf
);
62 void CDECL
strgen_error(const char *s
, ...)
67 vseprintf(buf
, lastof(buf
), s
, va
);
69 fprintf(stderr
, LINE_NUM_FMT("error"), _file
, _cur_line
, buf
);
73 void NORETURN CDECL
strgen_fatal(const char *s
, ...)
78 vseprintf(buf
, lastof(buf
), s
, va
);
80 fprintf(stderr
, LINE_NUM_FMT("FATAL"), _file
, _cur_line
, buf
);
82 fprintf(stderr
, LINE_NUM_FMT("warning"), _file
, _cur_line
, "language is not compiled");
84 throw std::exception();
87 void NORETURN CDECL
error(const char *s
, ...)
92 vseprintf(buf
, lastof(buf
), s
, va
);
94 fprintf(stderr
, LINE_NUM_FMT("FATAL"), _file
, _cur_line
, buf
);
96 fprintf(stderr
, LINE_NUM_FMT("warning"), _file
, _cur_line
, "language is not compiled");
101 /** A reader that simply reads using fopen. */
102 struct FileStringReader
: StringReader
{
103 FILE *fh
; ///< The file we are reading.
107 * @param data The data to fill during reading.
108 * @param file The file we are reading.
109 * @param master Are we reading the master file?
110 * @param translation Are we reading a translation?
112 FileStringReader(StringData
&data
, const char *file
, bool master
, bool translation
) :
113 StringReader(data
, file
, master
, translation
)
115 this->fh
= fopen(file
, "rb");
116 if (this->fh
== NULL
) error("Could not open %s", file
);
119 /** Free/close the file. */
120 virtual ~FileStringReader()
125 /* virtual */ char *ReadLine(char *buffer
, const char *last
)
127 return fgets(buffer
, ClampToU16(last
- buffer
+ 1), this->fh
);
130 /* virtual */ void HandlePragma(char *str
);
132 /* virtual */ void ParseFile()
134 this->StringReader::ParseFile();
136 if (StrEmpty(_lang
.name
) || StrEmpty(_lang
.own_name
) || StrEmpty(_lang
.isocode
)) {
137 error("Language must include ##name, ##ownname and ##isocode");
142 void FileStringReader::HandlePragma(char *str
)
144 if (!memcmp(str
, "id ", 3)) {
145 this->data
.next_string_id
= strtoul(str
+ 3, NULL
, 0);
146 } else if (!memcmp(str
, "name ", 5)) {
147 strecpy(_lang
.name
, str
+ 5, lastof(_lang
.name
));
148 } else if (!memcmp(str
, "ownname ", 8)) {
149 strecpy(_lang
.own_name
, str
+ 8, lastof(_lang
.own_name
));
150 } else if (!memcmp(str
, "isocode ", 8)) {
151 strecpy(_lang
.isocode
, str
+ 8, lastof(_lang
.isocode
));
152 } else if (!memcmp(str
, "textdir ", 8)) {
153 if (!memcmp(str
+ 8, "ltr", 3)) {
154 _lang
.text_dir
= TD_LTR
;
155 } else if (!memcmp(str
+ 8, "rtl", 3)) {
156 _lang
.text_dir
= TD_RTL
;
158 error("Invalid textdir %s", str
+ 8);
160 } else if (!memcmp(str
, "digitsep ", 9)) {
162 strecpy(_lang
.digit_group_separator
, strcmp(str
, "{NBSP}") == 0 ? NBSP
: str
, lastof(_lang
.digit_group_separator
));
163 } else if (!memcmp(str
, "digitsepcur ", 12)) {
165 strecpy(_lang
.digit_group_separator_currency
, strcmp(str
, "{NBSP}") == 0 ? NBSP
: str
, lastof(_lang
.digit_group_separator_currency
));
166 } else if (!memcmp(str
, "decimalsep ", 11)) {
168 strecpy(_lang
.digit_decimal_separator
, strcmp(str
, "{NBSP}") == 0 ? NBSP
: str
, lastof(_lang
.digit_decimal_separator
));
169 } else if (!memcmp(str
, "winlangid ", 10)) {
170 const char *buf
= str
+ 10;
171 long langid
= strtol(buf
, NULL
, 16);
172 if (langid
> (long)UINT16_MAX
|| langid
< 0) {
173 error("Invalid winlangid %s", buf
);
175 _lang
.winlangid
= (uint16
)langid
;
176 } else if (!memcmp(str
, "grflangid ", 10)) {
177 const char *buf
= str
+ 10;
178 long langid
= strtol(buf
, NULL
, 16);
179 if (langid
>= 0x7F || langid
< 0) {
180 error("Invalid grflangid %s", buf
);
182 _lang
.newgrflangid
= (uint8
)langid
;
183 } else if (!memcmp(str
, "gender ", 7)) {
184 if (this->master
) error("Genders are not allowed in the base translation.");
188 const char *s
= ParseWord(&buf
);
190 if (s
== NULL
) break;
191 if (_lang
.num_genders
>= MAX_NUM_GENDERS
) error("Too many genders, max %d", MAX_NUM_GENDERS
);
192 strecpy(_lang
.genders
[_lang
.num_genders
], s
, lastof(_lang
.genders
[_lang
.num_genders
]));
195 } else if (!memcmp(str
, "case ", 5)) {
196 if (this->master
) error("Cases are not allowed in the base translation.");
200 const char *s
= ParseWord(&buf
);
202 if (s
== NULL
) break;
203 if (_lang
.num_cases
>= MAX_NUM_CASES
) error("Too many cases, max %d", MAX_NUM_CASES
);
204 strecpy(_lang
.cases
[_lang
.num_cases
], s
, lastof(_lang
.cases
[_lang
.num_cases
]));
208 StringReader::HandlePragma(str
);
212 bool CompareFiles(const char *n1
, const char *n2
)
214 FILE *f2
= fopen(n2
, "rb");
215 if (f2
== NULL
) return false;
217 FILE *f1
= fopen(n1
, "rb");
218 if (f1
== NULL
) error("can't open %s", n1
);
224 l1
= fread(b1
, 1, sizeof(b1
), f1
);
225 l2
= fread(b2
, 1, sizeof(b2
), f2
);
227 if (l1
!= l2
|| memcmp(b1
, b2
, l1
)) {
239 /** Base class for writing data to disk. */
241 FILE *fh
; ///< The file handle we're writing to.
242 const char *filename
; ///< The file name we're writing to.
245 * Open a file to write to.
246 * @param filename The file to open.
248 FileWriter(const char *filename
)
250 this->filename
= stredup(filename
);
251 this->fh
= fopen(this->filename
, "wb");
253 if (this->fh
== NULL
) {
254 error("Could not open %s", this->filename
);
258 /** Finalise the writing. */
265 /** Make sure the file is closed. */
266 virtual ~FileWriter()
268 /* If we weren't closed an exception was thrown, so remove the temporary file. */
271 unlink(this->filename
);
273 free(this->filename
);
277 struct HeaderFileWriter
: HeaderWriter
, FileWriter
{
278 /** The real file name we eventually want to write to. */
279 const char *real_filename
;
280 /** The previous string ID that was printed. */
284 * Open a file to write to.
285 * @param filename The file to open.
287 HeaderFileWriter(const char *filename
) : FileWriter("tmp.xxx"),
288 real_filename(stredup(filename
)), prev(0)
290 fprintf(this->fh
, "/* This file is automatically generated. Do not modify */\n\n");
291 fprintf(this->fh
, "#ifndef TABLE_STRINGS_H\n");
292 fprintf(this->fh
, "#define TABLE_STRINGS_H\n");
295 /** Free the filename. */
301 void WriteStringID(const char *name
, int stringid
)
303 if (prev
+ 1 != stringid
) fprintf(this->fh
, "\n");
304 fprintf(this->fh
, "static const StringID %s = 0x%X;\n", name
, stringid
);
308 void Finalise(const StringData
&data
)
310 /* Find the plural form with the most amount of cases. */
311 int max_plural_forms
= 0;
312 for (uint i
= 0; i
< lengthof(_plural_forms
); i
++) {
313 max_plural_forms
= max(max_plural_forms
, _plural_forms
[i
].plural_count
);
318 "static const uint LANGUAGE_PACK_VERSION = 0x%X;\n"
319 "static const uint LANGUAGE_MAX_PLURAL = %d;\n"
320 "static const uint LANGUAGE_MAX_PLURAL_FORMS = %d;\n\n",
321 (uint
)data
.Version(), (uint
)lengthof(_plural_forms
), max_plural_forms
324 fprintf(this->fh
, "#endif /* TABLE_STRINGS_H */\n");
326 this->FileWriter::Finalise();
328 if (CompareFiles(this->filename
, this->real_filename
)) {
329 /* files are equal. tmp.xxx is not needed */
330 unlink(this->filename
);
332 /* else rename tmp.xxx into filename */
333 #if defined(WIN32) || defined(WIN64)
334 unlink(this->real_filename
);
336 if (rename(this->filename
, this->real_filename
) == -1) error("rename() failed");
341 /** Class for writing a language to disk. */
342 struct LanguageFileWriter
: LanguageWriter
, FileWriter
{
344 * Open a file to write to.
345 * @param filename The file to open.
347 LanguageFileWriter(const char *filename
) : FileWriter(filename
)
351 void WriteHeader(const LanguagePackHeader
*header
)
353 this->Write((const byte
*)header
, sizeof(*header
));
358 if (fputc(0, this->fh
) == EOF
) {
359 error("Could not write to %s", this->filename
);
361 this->FileWriter::Finalise();
364 void Write(const byte
*buffer
, size_t length
)
366 if (fwrite(buffer
, sizeof(*buffer
), length
, this->fh
) != length
) {
367 error("Could not write to %s", this->filename
);
372 /** Multi-OS mkdirectory function */
373 static inline void ottd_mkdir(const char *directory
)
375 /* Ignore directory creation errors; they'll surface later on, and most
376 * of the time they are 'directory already exists' errors anyhow. */
377 #if defined(WIN32) || defined(__WATCOMC__)
380 mkdir(directory
, 0755);
385 * Create a path consisting of an already existing path, a possible
386 * path separator and the filename. The separator is only appended if the path
387 * does not already end with a separator
389 static inline char *mkpath(char *buf
, const char *last
, const char *path
, const char *file
)
391 strecpy(buf
, path
, last
); // copy directory into buffer
393 char *p
= strchr(buf
, '\0'); // add path separator if necessary
394 if (p
[-1] != PATHSEPCHAR
&& p
!= last
) *p
++ = PATHSEPCHAR
;
395 strecpy(p
, file
, last
); // concatenate filename at end of buffer
399 #if defined(__MINGW32__)
401 * On MingW, it is common that both / as \ are accepted in the
402 * params. To go with those flow, we rewrite all incoming /
403 * simply to \, so internally we can safely assume \.
405 static inline char *replace_pathsep(char *s
)
407 for (char *c
= s
; *c
!= '\0'; c
++) if (*c
== '/') *c
= '\\';
411 static inline char *replace_pathsep(char *s
) { return s
; }
414 /** Options of strgen. */
415 static const OptionData _opts
[] = {
416 GETOPT_NOVAL( 'v', "--version"),
417 GETOPT_GENERAL('C', '\0', "-export-commands", ODF_NO_VALUE
),
418 GETOPT_GENERAL('L', '\0', "-export-plurals", ODF_NO_VALUE
),
419 GETOPT_GENERAL('P', '\0', "-export-pragmas", ODF_NO_VALUE
),
420 GETOPT_NOVAL( 't', "--todo"),
421 GETOPT_NOVAL( 'w', "--warning"),
422 GETOPT_NOVAL( 'h', "--help"),
423 GETOPT_GENERAL('h', '?', NULL
, ODF_NO_VALUE
),
424 GETOPT_VALUE( 's', "--source_dir"),
425 GETOPT_VALUE( 'd', "--dest_dir"),
429 int CDECL
main(int argc
, char *argv
[])
431 char pathbuf
[MAX_PATH
];
432 const char *src_dir
= ".";
433 const char *dest_dir
= NULL
;
435 GetOptData
mgo(argc
- 1, argv
+ 1, _opts
);
437 int i
= mgo
.GetOpt();
446 printf("args\tflags\tcommand\treplacement\n");
447 for (const CmdStruct
*cs
= _cmd_structs
; cs
< endof(_cmd_structs
); cs
++) {
449 if (cs
->proc
== EmitGender
) {
450 flags
= 'g'; // Command needs number of parameters defined by number of genders
451 } else if (cs
->proc
== EmitPlural
) {
452 flags
= 'p'; // Command needs number of parameters defined by plural value
453 } else if (cs
->flags
& C_DONTCOUNT
) {
454 flags
= 'i'; // Command may be in the translation when it is not in base
456 flags
= '0'; // Command needs no parameters
458 printf("%i\t%c\t\"%s\"\t\"%s\"\n", cs
->consumes
, flags
, cs
->cmd
, strstr(cs
->cmd
, "STRING") ? "STRING" : cs
->cmd
);
463 printf("count\tdescription\tnames\n");
464 for (const PluralForm
*pf
= _plural_forms
; pf
< endof(_plural_forms
); pf
++) {
465 printf("%i\t\"%s\"\t%s\n", pf
->plural_count
, pf
->description
, pf
->names
);
470 printf("name\tflags\tdefault\tdescription\n");
471 for (size_t i
= 0; i
< lengthof(_pragmas
); i
++) {
472 printf("\"%s\"\t%s\t\"%s\"\t\"%s\"\n",
473 _pragmas
[i
][0], _pragmas
[i
][1], _pragmas
[i
][2], _pragmas
[i
][3]);
487 "strgen - $Revision$\n"
488 " -v | --version print version information and exit\n"
489 " -t | --todo replace any untranslated strings with '<TODO>'\n"
490 " -w | --warning print a warning for any untranslated strings\n"
491 " -h | -? | --help print this help message and exit\n"
492 " -s | --source_dir search for english.txt in the specified directory\n"
493 " -d | --dest_dir put output file in the specified directory, create if needed\n"
494 " -export-commands export all commands and exit\n"
495 " -export-plurals export all plural forms and exit\n"
496 " -export-pragmas export all pragmas and exit\n"
497 " Run without parameters and strgen will search for english.txt and parse it,\n"
498 " creating strings.h. Passing an argument, strgen will translate that language\n"
499 " file using english.txt as a reference and output <language>.lng."
504 src_dir
= replace_pathsep(mgo
.opt
);
508 dest_dir
= replace_pathsep(mgo
.opt
);
512 fprintf(stderr
, "Invalid arguments\n");
517 if (dest_dir
== NULL
) dest_dir
= src_dir
; // if dest_dir is not specified, it equals src_dir
520 /* strgen has two modes of operation. If no (free) arguments are passed
521 * strgen generates strings.h to the destination directory. If it is supplied
522 * with a (free) parameter the program will translate that language to destination
523 * directory. As input english.txt is parsed from the source directory */
524 if (mgo
.numleft
== 0) {
525 mkpath(pathbuf
, lastof(pathbuf
), src_dir
, "english.txt");
527 /* parse master file */
528 StringData
data(TEXT_TAB_END
);
529 FileStringReader
master_reader(data
, pathbuf
, true, false);
530 master_reader
.ParseFile();
531 if (_errors
!= 0) return 1;
533 /* write strings.h */
534 ottd_mkdir(dest_dir
);
535 mkpath(pathbuf
, lastof(pathbuf
), dest_dir
, "strings.h");
537 HeaderFileWriter
writer(pathbuf
);
538 writer
.WriteHeader(data
);
539 writer
.Finalise(data
);
540 } else if (mgo
.numleft
>= 1) {
543 mkpath(pathbuf
, lastof(pathbuf
), src_dir
, "english.txt");
545 StringData
data(TEXT_TAB_END
);
546 /* parse master file and check if target file is correct */
547 FileStringReader
master_reader(data
, pathbuf
, true, false);
548 master_reader
.ParseFile();
550 for (int i
= 0; i
< mgo
.numleft
; i
++) {
551 data
.FreeTranslation();
553 const char *translation
= replace_pathsep(mgo
.argv
[i
]);
554 const char *file
= strrchr(translation
, PATHSEPCHAR
);
555 FileStringReader
translation_reader(data
, translation
, false, file
== NULL
|| strcmp(file
+ 1, "english.txt") != 0);
556 translation_reader
.ParseFile(); // target file
557 if (_errors
!= 0) return 1;
559 /* get the targetfile, strip any directories and append to destination path */
560 r
= strrchr(mgo
.argv
[i
], PATHSEPCHAR
);
561 mkpath(pathbuf
, lastof(pathbuf
), dest_dir
, (r
!= NULL
) ? &r
[1] : mgo
.argv
[i
]);
563 /* rename the .txt (input-extension) to .lng */
564 r
= strrchr(pathbuf
, '.');
565 if (r
== NULL
|| strcmp(r
, ".txt") != 0) r
= strchr(pathbuf
, '\0');
566 strecpy(r
, ".lng", lastof(pathbuf
));
568 LanguageFileWriter
writer(pathbuf
);
569 writer
.WriteLang(data
);
572 /* if showing warnings, print a summary of the language */
573 if ((_show_todo
& 2) != 0) {
574 fprintf(stdout
, "%d warnings and %d errors for %s\n", _warnings
, _errors
, pathbuf
);