Subversion Repositories SvarDOS

Rev

Rev 1290 | Rev 1298 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
1290 bernd.boec 1
/* This file is part of the svarlang project and is published under the terms
2
 * of the MIT license.
3
 *
4
 * Copyright (C) 2021-2023 Mateusz Viste
5
 *
6
 * Permission is hereby granted, free of charge, to any person obtaining a
7
 * copy of this software and associated documentation files (the "Software"),
8
 * to deal in the Software without restriction, including without limitation
9
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
10
 * and/or sell copies of the Software, and to permit persons to whom the
11
 * Software is furnished to do so, subject to the following conditions:
12
 *
13
 * The above copyright notice and this permission notice shall be included in
14
 * all copies or substantial portions of the Software.
15
 *
16
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
22
 * DEALINGS IN THE SOFTWARE.
23
 */
24
 
25
/* if WITHSTDIO is enabled, then remap file operations to use the standard
26
 * stdio amenities */
27
#ifdef WITHSTDIO
28
 
29
#include <stdio.h>   /* FILE, fopen(), fseek(), etc */
30
typedef FILE* FHANDLE;
31
#define FOPEN(x) fopen(x, "rb")
32
#define FCLOSE(x) fclose(x)
33
#define FSEEK(f,b) fseek(f,b,SEEK_CUR)
34
#define FREAD(f,t,b) fread(t, 1, b, f)
35
 
36
#else
37
 
38
#include <i86.h>
39
typedef unsigned short FHANDLE;
40
 
41
#endif
42
 
43
#include <stdlib.h>  /* NULL */
44
#include <string.h>  /* memcmp(), strcpy() */
45
 
46
#include "svarlang.h"
47
 
48
 
49
/* supplied through DEFLANG.C */
50
extern char svarlang_mem[];
51
extern unsigned short svarlang_dict[];
52
extern const unsigned short svarlang_memsz;
53
extern const unsigned short svarlang_string_count;
54
 
55
const char *svarlang_strid(unsigned short id) {
56
   size_t left = 0, right = svarlang_string_count - 1, x;
57
   unsigned short v;
58
 
59
   if (svarlang_string_count == 0) return "";
60
 
61
   while (left <= right ) {
62
      x = left + ( (right - left ) >> 2 );
63
      v = svarlang_dict[x * 2];
64
      if ( id == v )  {
65
        return svarlang_mem + svarlang_dict[x * 2 + 1];
66
      }
67
      else if ( id > v ) left = x + 1;
68
      else right = x - 1;
69
   }
70
   return "";
71
}
72
 
1293 mateusz.vi 73
 
1290 bernd.boec 74
/* routines below are simplified (dos-based) versions of the libc FILE-related
75
 * functions. Using them avoids a dependency on FILE, hence makes the binary
76
 * smaller if the application does not need to pull fopen() and friends */
77
#ifndef WITHSTDIO
78
static unsigned short FOPEN(const char *s) {
79
  unsigned short fname_seg = FP_SEG(s);
80
  unsigned short fname_off = FP_OFF(s);
81
  unsigned short res = 0; /* fd 0 is already used by stdout so it's a good error value */
82
  _asm {
83
    push dx
84
    push ds
85
 
86
    mov ax, fname_seg
87
    mov dx, fname_off
88
    mov ds, ax
89
    mov ax, 0x3d00  /* open file, read-only (fname at DS:DX) */
90
    int 0x21
91
    pop ds
92
    jc ERR
93
    mov res, ax
94
 
95
    ERR:
96
    pop dx
97
  }
98
 
99
  return(res);
100
}
101
 
102
 
103
static void FCLOSE(unsigned short handle) {
104
  _asm {
105
    push bx
106
 
107
    mov ah, 0x3e
108
    mov bx, handle
109
    int 0x21
110
 
111
    pop bx
112
  }
113
}
114
 
115
 
116
static unsigned short FREAD(unsigned short handle, void *buff, unsigned short bytes) {
117
  unsigned short buff_seg = FP_SEG(buff);
118
  unsigned short buff_off = FP_OFF(buff);
119
  unsigned short res = 0;
120
 
121
  _asm {
122
    push bx
123
    push cx
124
    push dx
125
 
126
    mov bx, handle
127
    mov cx, bytes
128
    mov dx, buff_off
129
    mov ax, buff_seg
130
    push ds
131
    mov ds, ax
132
    mov ah, 0x3f    /* read cx bytes from file handle bx to DS:DX */
133
    int 0x21
134
    pop ds
135
    jc ERR
136
 
137
    mov res, ax
138
    ERR:
139
 
140
    pop dx
141
    pop cx
142
    pop bx
143
  }
144
 
145
  return(res);
146
}
147
 
148
 
149
static void FSEEK(unsigned short handle, unsigned short bytes) {
150
  _asm {
151
    push bx
152
    push cx
153
    push dx
154
 
155
    mov ax, 0x4201  /* move file pointer from cur pos + CX:DX */
156
    mov bx, handle
157
    xor cx, cx
158
    mov dx, bytes
159
    int 0x21
160
 
161
    pop dx
162
    pop cx
163
    pop bx
164
  }
165
}
166
#endif
167
 
1293 mateusz.vi 168
 
1290 bernd.boec 169
int svarlang_load(const char *fname, const char *lang) {
170
  unsigned short langid;
1293 mateusz.vi 171
  unsigned long hdr;
1290 bernd.boec 172
  unsigned short buff16[2];
173
  FHANDLE fd;
174
  unsigned short string_count;
175
 
176
  langid = *((unsigned short *)lang);
177
  langid &= 0xDFDF; /* make sure lang is upcase */
178
 
179
  fd = FOPEN(fname);
180
  if (!fd) return(-1);
181
 
1293 mateusz.vi 182
  /* read hdr, should be "SvL\x1a" */
183
  if ((FREAD(fd, &hdr, 4) != 4) || (hdr != 0x1a4c7653L)) { /* 0x1a4c7653 = 'SvL\x1a' */
1290 bernd.boec 184
    FCLOSE(fd);
185
    return(-3);
186
  }
187
 
188
  /* read string count */
189
  if ((FREAD(fd, &string_count, 2) != 2) || (string_count != svarlang_string_count)) {
190
    FCLOSE(fd);
191
    return(-6);
192
  }
193
 
194
  /* read next lang id and string table size in file */
195
  while (FREAD(fd, buff16, 4) == 4) {
196
 
197
    /* is it the lang I am looking for? */
198
    if (buff16[0] != langid) { /* skip to next lang */
199
      FSEEK(fd, svarlang_string_count * 4);
200
      FSEEK(fd, buff16[1]);
201
      continue;
202
    }
203
 
204
    /* found - but do I have enough memory space? */
205
    if (buff16[1] >= svarlang_memsz) {
206
      FCLOSE(fd);
207
      return(-4);
208
    }
209
 
210
    /* load dictionary & strings */
211
    if ((FREAD(fd, svarlang_dict, svarlang_string_count * 4) != svarlang_string_count * 4) ||
212
       (FREAD(fd, svarlang_mem, buff16[1]) != buff16[1])) {
213
      FCLOSE(fd);
214
      return -7;
215
    }
216
    FCLOSE(fd);
217
    return(0);
218
  }
219
 
220
  FCLOSE(fd);
221
  return(-5);
222
}