blob: d3fef771f6ab579d3c42ff064a6a0d91d1c4afc5 [file] [log] [blame]
J. Duke319a3b92007-12-01 00:00:00 +00001/*
2 * Copyright 2003-2004 Sun Microsystems, Inc. All Rights Reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation. Sun designates this
8 * particular file as subject to the "Classpath" exception as provided
9 * by Sun in the LICENSE file that accompanied this code.
10 *
11 * This code is distributed in the hope that it will be useful, but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14 * version 2 for more details (a copy is included in the LICENSE file that
15 * accompanied this code).
16 *
17 * You should have received a copy of the GNU General Public License version
18 * 2 along with this work; if not, write to the Free Software Foundation,
19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20 *
21 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
22 * CA 95054 USA or visit www.sun.com if you need additional information or
23 * have any questions.
24 */
25
26package sun.nio.cs.ext;
27
28import java.nio.ByteBuffer;
29import java.nio.CharBuffer;
30import java.nio.charset.Charset;
31import java.nio.charset.CharsetDecoder;
32import java.nio.charset.CharsetEncoder;
33import java.nio.charset.CoderResult;
34import java.nio.charset.CharacterCodingException;
35import java.nio.charset.MalformedInputException;
36import sun.nio.cs.HistoricallyNamedCharset;
37import java.security.AccessController;
38import sun.security.action.GetPropertyAction;
39import static java.lang.Character.UnicodeBlock;
40
41
42public class JISAutoDetect
43 extends Charset
44 implements HistoricallyNamedCharset
45{
46
47 private final static int EUCJP_MASK = 0x01;
48 private final static int SJIS2B_MASK = 0x02;
49 private final static int SJIS1B_MASK = 0x04;
50 private final static int EUCJP_KANA1_MASK = 0x08;
51 private final static int EUCJP_KANA2_MASK = 0x10;
52
53 public JISAutoDetect() {
54 super("x-JISAutoDetect", ExtendedCharsets.aliasesFor("x-JISAutoDetect"));
55 }
56
57 public boolean contains(Charset cs) {
58 return ((cs.name().equals("US-ASCII"))
59 || (cs instanceof SJIS)
60 || (cs instanceof EUC_JP)
61 || (cs instanceof ISO2022_JP));
62 }
63
64 public boolean canEncode() {
65 return false;
66 }
67
68 public CharsetDecoder newDecoder() {
69 return new Decoder(this);
70 }
71
72 public String historicalName() {
73 return "JISAutoDetect";
74 }
75
76 public CharsetEncoder newEncoder() {
77 throw new UnsupportedOperationException();
78 }
79
80 /**
81 * accessor methods used to share byte masking tables
82 * with the sun.io JISAutoDetect implementation
83 */
84
85 public byte[] getByteMask1() {
86 return Decoder.maskTable1;
87 }
88
89 public byte[] getByteMask2() {
90 return Decoder.maskTable2;
91 }
92
93 public final static boolean canBeSJIS1B(int mask) {
94 return (mask & SJIS1B_MASK) != 0;
95 }
96
97 public final static boolean canBeEUCJP(int mask) {
98 return (mask & EUCJP_MASK) != 0;
99 }
100
101 public final static boolean canBeEUCKana(int mask1, int mask2) {
102 return ((mask1 & EUCJP_KANA1_MASK) != 0)
103 && ((mask2 & EUCJP_KANA2_MASK) != 0);
104 }
105
106 // A heuristic algorithm for guessing if EUC-decoded text really
107 // might be Japanese text. Better heuristics are possible...
108 private static boolean looksLikeJapanese(CharBuffer cb) {
109 int hiragana = 0; // Fullwidth Hiragana
110 int katakana = 0; // Halfwidth Katakana
111 while (cb.hasRemaining()) {
112 char c = cb.get();
113 if (0x3040 <= c && c <= 0x309f && ++hiragana > 1) return true;
114 if (0xff65 <= c && c <= 0xff9f && ++katakana > 1) return true;
115 }
116 return false;
117 }
118
119 private static class Decoder extends CharsetDecoder {
120
121 private final static String SJISName = getSJISName();
122 private final static String EUCJPName = getEUCJPName();
123 private DelegatableDecoder detectedDecoder = null;
124
125 public Decoder(Charset cs) {
126 super(cs, 0.5f, 1.0f);
127 }
128
129 private static boolean isPlainASCII(byte b) {
130 return b >= 0 && b != 0x1b;
131 }
132
133 private static void copyLeadingASCII(ByteBuffer src, CharBuffer dst) {
134 int start = src.position();
135 int limit = start + Math.min(src.remaining(), dst.remaining());
136 int p;
137 byte b;
138 for (p = start; p < limit && isPlainASCII(b = src.get(p)); p++)
139 dst.put((char)(b & 0xff));
140 src.position(p);
141 }
142
143 private CoderResult decodeLoop(Charset cs,
144 ByteBuffer src, CharBuffer dst) {
145 detectedDecoder = (DelegatableDecoder) cs.newDecoder();
146 return detectedDecoder.decodeLoop(src, dst);
147 }
148
149 protected CoderResult decodeLoop(ByteBuffer src, CharBuffer dst) {
150 if (detectedDecoder == null) {
151 copyLeadingASCII(src, dst);
152
153 // All ASCII?
154 if (! src.hasRemaining())
155 return CoderResult.UNDERFLOW;
156 if (! dst.hasRemaining())
157 return CoderResult.OVERFLOW;
158
159 // We need to perform double, not float, arithmetic; otherwise
160 // we lose low order bits when src is larger than 2**24.
161 int cbufsiz = (int)(src.limit() * (double)maxCharsPerByte());
162 CharBuffer sandbox = CharBuffer.allocate(cbufsiz);
163
164 // First try ISO-2022-JP, since there is no ambiguity
165 Charset cs2022 = Charset.forName("ISO-2022-JP");
166 DelegatableDecoder dd2022
167 = (DelegatableDecoder) cs2022.newDecoder();
168 ByteBuffer src2022 = src.asReadOnlyBuffer();
169 CoderResult res2022 = dd2022.decodeLoop(src2022, sandbox);
170 if (! res2022.isError())
171 return decodeLoop(cs2022, src, dst);
172
173 // We must choose between EUC and SJIS
174 Charset csEUCJ = Charset.forName(EUCJPName);
175 Charset csSJIS = Charset.forName(SJISName);
176
177 DelegatableDecoder ddEUCJ
178 = (DelegatableDecoder) csEUCJ.newDecoder();
179 ByteBuffer srcEUCJ = src.asReadOnlyBuffer();
180 sandbox.clear();
181 CoderResult resEUCJ = ddEUCJ.decodeLoop(srcEUCJ, sandbox);
182 // If EUC decoding fails, must be SJIS
183 if (resEUCJ.isError())
184 return decodeLoop(csSJIS, src, dst);
185
186 DelegatableDecoder ddSJIS
187 = (DelegatableDecoder) csSJIS.newDecoder();
188 ByteBuffer srcSJIS = src.asReadOnlyBuffer();
189 CharBuffer sandboxSJIS = CharBuffer.allocate(cbufsiz);
190 CoderResult resSJIS = ddSJIS.decodeLoop(srcSJIS, sandboxSJIS);
191 // If SJIS decoding fails, must be EUC
192 if (resSJIS.isError())
193 return decodeLoop(csEUCJ, src, dst);
194
195 // From here on, we have some ambiguity, and must guess.
196
197 // We prefer input that does not appear to end mid-character.
198 if (srcEUCJ.position() > srcSJIS.position())
199 return decodeLoop(csEUCJ, src, dst);
200
201 if (srcEUCJ.position() < srcSJIS.position())
202 return decodeLoop(csSJIS, src, dst);
203
204 // end-of-input is after the first byte of the first char?
205 if (src.position() == srcEUCJ.position())
206 return CoderResult.UNDERFLOW;
207
208 // Use heuristic knowledge of typical Japanese text
209 sandbox.flip();
210 Charset guess = looksLikeJapanese(sandbox) ? csEUCJ : csSJIS;
211 return decodeLoop(guess, src, dst);
212 }
213
214 return detectedDecoder.decodeLoop(src, dst);
215 }
216
217 protected void implReset() {
218 detectedDecoder = null;
219 }
220
221 protected CoderResult implFlush(CharBuffer out) {
222 if (detectedDecoder != null)
223 return detectedDecoder.implFlush(out);
224 else
225 return super.implFlush(out);
226 }
227
228 public boolean isAutoDetecting() {
229 return true;
230 }
231
232 public boolean isCharsetDetected() {
233 return detectedDecoder != null;
234 }
235
236 public Charset detectedCharset() {
237 if (detectedDecoder == null)
238 throw new IllegalStateException("charset not yet detected");
239 return ((CharsetDecoder) detectedDecoder).charset();
240 }
241
242 /**
243 * Returned Shift_JIS Charset name is OS dependent
244 */
245 private static String getSJISName() {
246 String osName = AccessController.doPrivileged(
247 new GetPropertyAction("os.name"));
248 if (osName.equals("Solaris") || osName.equals("SunOS"))
249 return("PCK");
250 else if (osName.startsWith("Windows"))
251 return("windows-31J");
252 else
253 return("Shift_JIS");
254 }
255
256 /**
257 * Returned EUC-JP Charset name is OS dependent
258 */
259
260 private static String getEUCJPName() {
261 String osName = AccessController.doPrivileged(
262 new GetPropertyAction("os.name"));
263 if (osName.equals("Solaris") || osName.equals("SunOS"))
264 return("x-eucjp-open");
265 else
266 return("EUC_JP");
267 }
268
269 // Mask tables - each entry indicates possibility of first or
270 // second byte being SJIS or EUC_JP
271 private static final byte maskTable1[] = {
272 0, 0, 0, 0, // 0x00 - 0x03
273 0, 0, 0, 0, // 0x04 - 0x07
274 0, 0, 0, 0, // 0x08 - 0x0b
275 0, 0, 0, 0, // 0x0c - 0x0f
276 0, 0, 0, 0, // 0x10 - 0x13
277 0, 0, 0, 0, // 0x14 - 0x17
278 0, 0, 0, 0, // 0x18 - 0x1b
279 0, 0, 0, 0, // 0x1c - 0x1f
280 0, 0, 0, 0, // 0x20 - 0x23
281 0, 0, 0, 0, // 0x24 - 0x27
282 0, 0, 0, 0, // 0x28 - 0x2b
283 0, 0, 0, 0, // 0x2c - 0x2f
284 0, 0, 0, 0, // 0x30 - 0x33
285 0, 0, 0, 0, // 0x34 - 0x37
286 0, 0, 0, 0, // 0x38 - 0x3b
287 0, 0, 0, 0, // 0x3c - 0x3f
288 0, 0, 0, 0, // 0x40 - 0x43
289 0, 0, 0, 0, // 0x44 - 0x47
290 0, 0, 0, 0, // 0x48 - 0x4b
291 0, 0, 0, 0, // 0x4c - 0x4f
292 0, 0, 0, 0, // 0x50 - 0x53
293 0, 0, 0, 0, // 0x54 - 0x57
294 0, 0, 0, 0, // 0x58 - 0x5b
295 0, 0, 0, 0, // 0x5c - 0x5f
296 0, 0, 0, 0, // 0x60 - 0x63
297 0, 0, 0, 0, // 0x64 - 0x67
298 0, 0, 0, 0, // 0x68 - 0x6b
299 0, 0, 0, 0, // 0x6c - 0x6f
300 0, 0, 0, 0, // 0x70 - 0x73
301 0, 0, 0, 0, // 0x74 - 0x77
302 0, 0, 0, 0, // 0x78 - 0x7b
303 0, 0, 0, 0, // 0x7c - 0x7f
304 0, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x80 - 0x83
305 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x84 - 0x87
306 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x88 - 0x8b
307 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, // 0x8c - 0x8f
308 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x90 - 0x93
309 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x94 - 0x97
310 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x98 - 0x9b
311 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x9c - 0x9f
312 0, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xa0 - 0xa3
313 SJIS1B_MASK|EUCJP_MASK|EUCJP_KANA1_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xa4 - 0xa7
314 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xa8 - 0xab
315 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xac - 0xaf
316 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xb0 - 0xb3
317 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xb4 - 0xb7
318 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xb8 - 0xbb
319 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xbc - 0xbf
320 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xc0 - 0xc3
321 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xc4 - 0xc7
322 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xc8 - 0xcb
323 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xcc - 0xcf
324 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xd0 - 0xd3
325 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xd4 - 0xd7
326 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xd8 - 0xdb
327 SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, SJIS1B_MASK|EUCJP_MASK, // 0xdc - 0xdf
328 SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, // 0xe0 - 0xe3
329 SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, // 0xe4 - 0xe7
330 SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, // 0xe8 - 0xeb
331 SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, // 0xec - 0xef
332 SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, // 0xf0 - 0xf3
333 SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, // 0xf4 - 0xf7
334 SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, // 0xf8 - 0xfb
335 SJIS2B_MASK|EUCJP_MASK, EUCJP_MASK, EUCJP_MASK, 0 // 0xfc - 0xff
336 };
337
338 private static final byte maskTable2[] = {
339 0, 0, 0, 0, // 0x00 - 0x03
340 0, 0, 0, 0, // 0x04 - 0x07
341 0, 0, 0, 0, // 0x08 - 0x0b
342 0, 0, 0, 0, // 0x0c - 0x0f
343 0, 0, 0, 0, // 0x10 - 0x13
344 0, 0, 0, 0, // 0x14 - 0x17
345 0, 0, 0, 0, // 0x18 - 0x1b
346 0, 0, 0, 0, // 0x1c - 0x1f
347 0, 0, 0, 0, // 0x20 - 0x23
348 0, 0, 0, 0, // 0x24 - 0x27
349 0, 0, 0, 0, // 0x28 - 0x2b
350 0, 0, 0, 0, // 0x2c - 0x2f
351 0, 0, 0, 0, // 0x30 - 0x33
352 0, 0, 0, 0, // 0x34 - 0x37
353 0, 0, 0, 0, // 0x38 - 0x3b
354 0, 0, 0, 0, // 0x3c - 0x3f
355 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x40 - 0x43
356 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x44 - 0x47
357 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x48 - 0x4b
358 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x4c - 0x4f
359 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x50 - 0x53
360 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x54 - 0x57
361 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x58 - 0x5b
362 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x5c - 0x5f
363 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x60 - 0x63
364 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x64 - 0x67
365 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x68 - 0x6b
366 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x6c - 0x6f
367 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x70 - 0x73
368 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x74 - 0x77
369 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x78 - 0x7b
370 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, 0, // 0x7c - 0x7f
371 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x80 - 0x83
372 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x84 - 0x87
373 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x88 - 0x8b
374 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x8c - 0x8f
375 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x90 - 0x93
376 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x94 - 0x97
377 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x98 - 0x9b
378 SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, SJIS2B_MASK, // 0x9c - 0x9f
379 SJIS2B_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xa0 - 0xa3
380 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xa4 - 0xa7
381 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xa8 - 0xab
382 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xac - 0xaf
383 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xb0 - 0xb3
384 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xb4 - 0xb7
385 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xb8 - 0xbb
386 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xbc - 0xbf
387 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xc0 - 0xc3
388 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xc4 - 0xc7
389 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xc8 - 0xcb
390 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xcc - 0xcf
391 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xd0 - 0xd3
392 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xd4 - 0xd7
393 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xd8 - 0xdb
394 SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS1B_MASK|SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xdc - 0xdf
395 SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xe0 - 0xe3
396 SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xe4 - 0xe7
397 SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xe8 - 0xeb
398 SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xec - 0xef
399 SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, SJIS2B_MASK|EUCJP_MASK|EUCJP_KANA2_MASK, // 0xf0 - 0xf3
400 SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, // 0xf4 - 0xf7
401 SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, SJIS2B_MASK|EUCJP_MASK, // 0xf8 - 0xfb
402 SJIS2B_MASK|EUCJP_MASK, EUCJP_MASK, EUCJP_MASK, 0 // 0xfc - 0xff
403 };
404 }
405}