blob: 715f83f67296875a8c64bafea76d8dc9d8045ea6 [file] [log] [blame]
/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2002 Daniel Stenberg
*
* 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 software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
****************************************************************************/
#include <stdio.h>
#include "file.h"
#include "language.h"
#include "lang.h"
#include "debug.h"
#include "string.h"
#ifdef HAVE_LCD_BITMAP
#include "viewport.h"
#endif
/* The following header is generated by the build system and only defines
MAX_LANGUAGE_SIZE to be the size of the largest currently available
language! */
#include "max_language_size.h"
/* These defines must match the initial bytes in the binary lang file */
/* See tools/genlang (TODO: Use common include for both) */
#define LANGUAGE_COOKIE 0x1a
#define LANGUAGE_VERSION 0x06
#define LANGUAGE_FLAG_RTL 0x01
#define HEADER_SIZE 4
#define SUBHEADER_SIZE 6
static unsigned char language_buffer[MAX_LANGUAGE_SIZE];
static unsigned char lang_options = 0;
void lang_init(const unsigned char *builtin, unsigned char **dest, int count)
{
while(count--) {
*dest++ = (unsigned char *)builtin;
/* advance pointer to next string */
builtin += strlen((char *)builtin) + 1;
}
}
int lang_load(const char *filename, const unsigned char *builtin,
unsigned char **dest, unsigned char *buffer,
unsigned int user_num, int max_lang_size,
unsigned int max_id)
{
int lang_size;
int fd = open(filename, O_RDONLY);
int retcode=0;
unsigned char lang_header[HEADER_SIZE];
unsigned char sub_header[SUBHEADER_SIZE];
unsigned int id, foffset;
if(fd < 0)
return 1;
read(fd, lang_header, HEADER_SIZE);
if((lang_header[0] == LANGUAGE_COOKIE) &&
(lang_header[1] == LANGUAGE_VERSION) &&
(lang_header[2] == TARGET_ID)) {
/* jump to the proper entry in the table of subheaders */
lseek(fd, user_num * SUBHEADER_SIZE, SEEK_CUR);
read(fd, sub_header, SUBHEADER_SIZE);
/* read in information about the requested lang */
#if 0 /* unused */
unsigned int num_strings = (sub_header[0]<<8) | sub_header[1];
#endif
lang_size = (sub_header[2]<<8) | sub_header[3];
foffset = (sub_header[4]<<8) | sub_header[5];
if(lang_size <= max_lang_size) {
/* initialize with builtin */
lang_init(builtin, dest, max_id);
lseek(fd, foffset, SEEK_SET);
read(fd, buffer, lang_size);
while(lang_size>3) {
id = ((buffer[0]<<8) | buffer[1]); /* get two-byte id */
buffer += 2; /* pass the id */
if(id < max_id) {
#if 0
DEBUGF("%2x New: %30s ", id, buffer);
DEBUGF("Replaces: %s\n", dest[id]);
#endif
dest[id] = buffer; /* point to this string */
}
while(*buffer) { /* pass the string */
lang_size--;
buffer++;
}
lang_size-=3; /* the id and the terminating zero */
buffer++; /* pass the terminating zero-byte */
}
}
else {
DEBUGF("Language %s too large: %d\n", filename, lang_size);
retcode = 2;
}
}
else {
DEBUGF("Illegal language file\n");
retcode = 3;
}
close(fd);
lang_options = retcode ? 0 : lang_header[3];
return retcode;
}
int lang_core_load(const char *filename)
{
return lang_load(filename, core_language_builtin, language_strings,
language_buffer, 0, MAX_LANGUAGE_SIZE,
LANG_LAST_INDEX_IN_ARRAY);
}
int lang_english_to_id(const char *english)
{
int i;
unsigned char *ptr = (unsigned char *) core_language_builtin;
for (i = 0; i < LANG_LAST_INDEX_IN_ARRAY; i++) {
if (!strcmp(ptr, english))
return i;
ptr += strlen((char *)ptr) + 1; /* advance pointer to next string */
}
return -1;
}
int lang_is_rtl(void)
{
return (lang_options & LANGUAGE_FLAG_RTL) != 0;
}