11da177e4SLinus Torvalds /*
2f30c2269SUwe Zeisberger * linux/fs/nls/nls_euc-jp.c
31da177e4SLinus Torvalds *
41da177e4SLinus Torvalds * Added `OSF/JVC Recommended Code Set Conversion Specification
51da177e4SLinus Torvalds * between Japanese EUC and Shift-JIS' support: <hirofumi@mail.parknet.co.jp>
61da177e4SLinus Torvalds * (http://www.opengroup.or.jp/jvc/cde/sjis-euc-e.html)
71da177e4SLinus Torvalds */
81da177e4SLinus Torvalds
91da177e4SLinus Torvalds #include <linux/module.h>
101da177e4SLinus Torvalds #include <linux/kernel.h>
111da177e4SLinus Torvalds #include <linux/string.h>
121da177e4SLinus Torvalds #include <linux/nls.h>
131da177e4SLinus Torvalds #include <linux/errno.h>
141da177e4SLinus Torvalds
151da177e4SLinus Torvalds static struct nls_table *p_nls;
161da177e4SLinus Torvalds
171da177e4SLinus Torvalds #define IS_SJIS_LOW_BYTE(l) ((0x40 <= (l)) && ((l) <= 0xFC) && ((l) != 0x7F))
181da177e4SLinus Torvalds /* JIS X 0208 (include NEC spesial characters) */
191da177e4SLinus Torvalds #define IS_SJIS_JISX0208(h, l) ((((0x81 <= (h)) && ((h) <= 0x9F)) \
201da177e4SLinus Torvalds || ((0xE0 <= (h)) && ((h) <= 0xEA))) \
211da177e4SLinus Torvalds && IS_SJIS_LOW_BYTE(l))
221da177e4SLinus Torvalds #define IS_SJIS_JISX0201KANA(c) ((0xA1 <= (c)) && ((c) <= 0xDF))
231da177e4SLinus Torvalds #define IS_SJIS_UDC_LOW(h, l) (((0xF0 <= (h)) && ((h) <= 0xF4)) \
241da177e4SLinus Torvalds && IS_SJIS_LOW_BYTE(l))
251da177e4SLinus Torvalds #define IS_SJIS_UDC_HI(h, l) (((0xF5 <= (h)) && ((h) <= 0xF9)) \
261da177e4SLinus Torvalds && IS_SJIS_LOW_BYTE(l))
271da177e4SLinus Torvalds #define IS_SJIS_IBM(h, l) (((0xFA <= (h)) && ((h) <= 0xFC)) \
281da177e4SLinus Torvalds && IS_SJIS_LOW_BYTE(l))
291da177e4SLinus Torvalds #define IS_SJIS_NECIBM(h, l) (((0xED <= (h)) && ((h) <= 0xEE)) \
301da177e4SLinus Torvalds && IS_SJIS_LOW_BYTE(l))
311da177e4SLinus Torvalds #define MAP_SJIS2EUC(sjis_hi, sjis_lo, sjis_p, euc_hi, euc_lo, euc_p) { \
321da177e4SLinus Torvalds if ((sjis_lo) >= 0x9F) { \
331da177e4SLinus Torvalds (euc_hi) = (sjis_hi) * 2 - (((sjis_p) * 2 - (euc_p)) - 1); \
341da177e4SLinus Torvalds (euc_lo) = (sjis_lo) + 2; \
351da177e4SLinus Torvalds } else { \
361da177e4SLinus Torvalds (euc_hi) = (sjis_hi) * 2 - ((sjis_p) * 2 - (euc_p)); \
371da177e4SLinus Torvalds (euc_lo) = (sjis_lo) + ((sjis_lo) >= 0x7F ? 0x60 : 0x61); \
381da177e4SLinus Torvalds } \
391da177e4SLinus Torvalds } while(0)
401da177e4SLinus Torvalds
411da177e4SLinus Torvalds #define SS2 (0x8E) /* Single Shift 2 */
421da177e4SLinus Torvalds #define SS3 (0x8F) /* Single Shift 3 */
431da177e4SLinus Torvalds #define IS_EUC_BYTE(c) ((0xA1 <= (c)) && ((c) <= 0xFE))
441da177e4SLinus Torvalds #define IS_EUC_JISX0208(h, l) (IS_EUC_BYTE(h) && IS_EUC_BYTE(l))
451da177e4SLinus Torvalds #define IS_EUC_JISX0201KANA(h, l) (((h) == SS2) && (0xA1 <= (l) && (l) <= 0xDF))
461da177e4SLinus Torvalds #define IS_EUC_UDC_LOW(h, l) (((0xF5 <= (h)) && ((h) <= 0xFE)) \
471da177e4SLinus Torvalds && IS_EUC_BYTE(l))
481da177e4SLinus Torvalds #define IS_EUC_UDC_HI(h, l) IS_EUC_UDC_LOW(h, l) /* G3 block */
491da177e4SLinus Torvalds #define MAP_EUC2SJIS(euc_hi, euc_lo, euc_p, sjis_hi, sjis_lo, sjis_p) { \
501da177e4SLinus Torvalds if ((euc_hi) & 1) { \
511da177e4SLinus Torvalds (sjis_hi) = (euc_hi) / 2 + ((sjis_p) - (euc_p) / 2); \
521da177e4SLinus Torvalds (sjis_lo) = (euc_lo) - ((euc_lo) >= 0xE0 ? 0x60 : 0x61); \
531da177e4SLinus Torvalds } else { \
541da177e4SLinus Torvalds (sjis_hi) = (euc_hi) / 2 + (((sjis_p) - (euc_p) / 2) - 1); \
551da177e4SLinus Torvalds (sjis_lo) = (euc_lo) - 2; \
561da177e4SLinus Torvalds } \
571da177e4SLinus Torvalds } while(0)
581da177e4SLinus Torvalds
591da177e4SLinus Torvalds /* SJIS IBM extended characters to EUC map */
60*b9ec0339SDenys Vlasenko static const unsigned char sjisibm2euc_map[][2] = {
611da177e4SLinus Torvalds {0xF3, 0xF3}, {0xF3, 0xF4}, {0xF3, 0xF5}, {0xF3, 0xF6}, {0xF3, 0xF7},
621da177e4SLinus Torvalds {0xF3, 0xF8}, {0xF3, 0xF9}, {0xF3, 0xFA}, {0xF3, 0xFB}, {0xF3, 0xFC},
631da177e4SLinus Torvalds {0xF3, 0xFD}, {0xF3, 0xFE}, {0xF4, 0xA1}, {0xF4, 0xA2}, {0xF4, 0xA3},
641da177e4SLinus Torvalds {0xF4, 0xA4}, {0xF4, 0xA5}, {0xF4, 0xA6}, {0xF4, 0xA7}, {0xF4, 0xA8},
651da177e4SLinus Torvalds {0xA2, 0xCC}, {0xA2, 0xC3}, {0xF4, 0xA9}, {0xF4, 0xAA}, {0xF4, 0xAB},
661da177e4SLinus Torvalds {0xF4, 0xAC}, {0xF4, 0xAD}, {0xA2, 0xE8}, {0xD4, 0xE3}, {0xDC, 0xDF},
671da177e4SLinus Torvalds {0xE4, 0xE9}, {0xE3, 0xF8}, {0xD9, 0xA1}, {0xB1, 0xBB}, {0xF4, 0xAE},
681da177e4SLinus Torvalds {0xC2, 0xAD}, {0xC3, 0xFC}, {0xE4, 0xD0}, {0xC2, 0xBF}, {0xBC, 0xF4},
691da177e4SLinus Torvalds {0xB0, 0xA9}, {0xB0, 0xC8}, {0xF4, 0xAF}, {0xB0, 0xD2}, {0xB0, 0xD4},
701da177e4SLinus Torvalds {0xB0, 0xE3}, {0xB0, 0xEE}, {0xB1, 0xA7}, {0xB1, 0xA3}, {0xB1, 0xAC},
711da177e4SLinus Torvalds {0xB1, 0xA9}, {0xB1, 0xBE}, {0xB1, 0xDF}, {0xB1, 0xD8}, {0xB1, 0xC8},
721da177e4SLinus Torvalds {0xB1, 0xD7}, {0xB1, 0xE3}, {0xB1, 0xF4}, {0xB1, 0xE1}, {0xB2, 0xA3},
731da177e4SLinus Torvalds {0xF4, 0xB0}, {0xB2, 0xBB}, {0xB2, 0xE6}, {0x00, 0x00}, {0xB2, 0xED},
741da177e4SLinus Torvalds {0xB2, 0xF5}, {0xB2, 0xFC}, {0xF4, 0xB1}, {0xB3, 0xB5}, {0xB3, 0xD8},
751da177e4SLinus Torvalds {0xB3, 0xDB}, {0xB3, 0xE5}, {0xB3, 0xEE}, {0xB3, 0xFB}, {0xF4, 0xB2},
761da177e4SLinus Torvalds {0xF4, 0xB3}, {0xB4, 0xC0}, {0xB4, 0xC7}, {0xB4, 0xD0}, {0xB4, 0xDE},
771da177e4SLinus Torvalds {0xF4, 0xB4}, {0xB5, 0xAA}, {0xF4, 0xB5}, {0xB5, 0xAF}, {0xB5, 0xC4},
781da177e4SLinus Torvalds {0xB5, 0xE8}, {0xF4, 0xB6}, {0xB7, 0xC2}, {0xB7, 0xE4}, {0xB7, 0xE8},
791da177e4SLinus Torvalds {0xB7, 0xE7}, {0xF4, 0xB7}, {0xF4, 0xB8}, {0xF4, 0xB9}, {0xB8, 0xCE},
801da177e4SLinus Torvalds {0xB8, 0xE1}, {0xB8, 0xF5}, {0xB8, 0xF7}, {0xB8, 0xF8}, {0xB8, 0xFC},
811da177e4SLinus Torvalds {0xB9, 0xAF}, {0xB9, 0xB7}, {0xBA, 0xBE}, {0xBA, 0xDB}, {0xCD, 0xAA},
821da177e4SLinus Torvalds {0xBA, 0xE1}, {0xF4, 0xBA}, {0xBA, 0xEB}, {0xBB, 0xB3}, {0xBB, 0xB8},
831da177e4SLinus Torvalds {0xF4, 0xBB}, {0xBB, 0xCA}, {0xF4, 0xBC}, {0xF4, 0xBD}, {0xBB, 0xD0},
841da177e4SLinus Torvalds {0xBB, 0xDE}, {0xBB, 0xF4}, {0xBB, 0xF5}, {0xBB, 0xF9}, {0xBC, 0xE4},
851da177e4SLinus Torvalds {0xBC, 0xED}, {0xBC, 0xFE}, {0xF4, 0xBE}, {0xBD, 0xC2}, {0xBD, 0xE7},
861da177e4SLinus Torvalds {0xF4, 0xBF}, {0xBD, 0xF0}, {0xBE, 0xB0}, {0xBE, 0xAC}, {0xF4, 0xC0},
871da177e4SLinus Torvalds {0xBE, 0xB3}, {0xBE, 0xBD}, {0xBE, 0xCD}, {0xBE, 0xC9}, {0xBE, 0xE4},
881da177e4SLinus Torvalds {0xBF, 0xA8}, {0xBF, 0xC9}, {0xC0, 0xC4}, {0xC0, 0xE4}, {0xC0, 0xF4},
891da177e4SLinus Torvalds {0xC1, 0xA6}, {0xF4, 0xC1}, {0xC1, 0xF5}, {0xC1, 0xFC}, {0xF4, 0xC2},
901da177e4SLinus Torvalds {0xC1, 0xF8}, {0xC2, 0xAB}, {0xC2, 0xA1}, {0xC2, 0xA5}, {0xF4, 0xC3},
911da177e4SLinus Torvalds {0xC2, 0xB8}, {0xC2, 0xBA}, {0xF4, 0xC4}, {0xC2, 0xC4}, {0xC2, 0xD2},
921da177e4SLinus Torvalds {0xC2, 0xD7}, {0xC2, 0xDB}, {0xC2, 0xDE}, {0xC2, 0xED}, {0xC2, 0xF0},
931da177e4SLinus Torvalds {0xF4, 0xC5}, {0xC3, 0xA1}, {0xC3, 0xB5}, {0xC3, 0xC9}, {0xC3, 0xB9},
941da177e4SLinus Torvalds {0xF4, 0xC6}, {0xC3, 0xD8}, {0xC3, 0xFE}, {0xF4, 0xC7}, {0xC4, 0xCC},
951da177e4SLinus Torvalds {0xF4, 0xC8}, {0xC4, 0xD9}, {0xC4, 0xEA}, {0xC4, 0xFD}, {0xF4, 0xC9},
961da177e4SLinus Torvalds {0xC5, 0xA7}, {0xC5, 0xB5}, {0xC5, 0xB6}, {0xF4, 0xCA}, {0xC5, 0xD5},
971da177e4SLinus Torvalds {0xC6, 0xB8}, {0xC6, 0xD7}, {0xC6, 0xE0}, {0xC6, 0xEA}, {0xC6, 0xE3},
981da177e4SLinus Torvalds {0xC7, 0xA1}, {0xC7, 0xAB}, {0xC7, 0xC7}, {0xC7, 0xC3}, {0xC7, 0xCB},
991da177e4SLinus Torvalds {0xC7, 0xCF}, {0xC7, 0xD9}, {0xF4, 0xCB}, {0xF4, 0xCC}, {0xC7, 0xE6},
1001da177e4SLinus Torvalds {0xC7, 0xEE}, {0xC7, 0xFC}, {0xC7, 0xEB}, {0xC7, 0xF0}, {0xC8, 0xB1},
1011da177e4SLinus Torvalds {0xC8, 0xE5}, {0xC8, 0xF8}, {0xC9, 0xA6}, {0xC9, 0xAB}, {0xC9, 0xAD},
1021da177e4SLinus Torvalds {0xF4, 0xCD}, {0xC9, 0xCA}, {0xC9, 0xD3}, {0xC9, 0xE9}, {0xC9, 0xE3},
1031da177e4SLinus Torvalds {0xC9, 0xFC}, {0xC9, 0xF4}, {0xC9, 0xF5}, {0xF4, 0xCE}, {0xCA, 0xB3},
1041da177e4SLinus Torvalds {0xCA, 0xBD}, {0xCA, 0xEF}, {0xCA, 0xF1}, {0xCB, 0xAE}, {0xF4, 0xCF},
1051da177e4SLinus Torvalds {0xCB, 0xCA}, {0xCB, 0xE6}, {0xCB, 0xEA}, {0xCB, 0xF0}, {0xCB, 0xF4},
1061da177e4SLinus Torvalds {0xCB, 0xEE}, {0xCC, 0xA5}, {0xCB, 0xF9}, {0xCC, 0xAB}, {0xCC, 0xAE},
1071da177e4SLinus Torvalds {0xCC, 0xAD}, {0xCC, 0xB2}, {0xCC, 0xC2}, {0xCC, 0xD0}, {0xCC, 0xD9},
1081da177e4SLinus Torvalds {0xF4, 0xD0}, {0xCD, 0xBB}, {0xF4, 0xD1}, {0xCE, 0xBB}, {0xF4, 0xD2},
1091da177e4SLinus Torvalds {0xCE, 0xBA}, {0xCE, 0xC3}, {0xF4, 0xD3}, {0xCE, 0xF2}, {0xB3, 0xDD},
1101da177e4SLinus Torvalds {0xCF, 0xD5}, {0xCF, 0xE2}, {0xCF, 0xE9}, {0xCF, 0xED}, {0xF4, 0xD4},
1111da177e4SLinus Torvalds {0xF4, 0xD5}, {0xF4, 0xD6}, {0x00, 0x00}, {0xF4, 0xD7}, {0xD0, 0xE5},
1121da177e4SLinus Torvalds {0xF4, 0xD8}, {0xD0, 0xE9}, {0xD1, 0xE8}, {0xF4, 0xD9}, {0xF4, 0xDA},
1131da177e4SLinus Torvalds {0xD1, 0xEC}, {0xD2, 0xBB}, {0xF4, 0xDB}, {0xD3, 0xE1}, {0xD3, 0xE8},
1141da177e4SLinus Torvalds {0xD4, 0xA7}, {0xF4, 0xDC}, {0xF4, 0xDD}, {0xD4, 0xD4}, {0xD4, 0xF2},
1151da177e4SLinus Torvalds {0xD5, 0xAE}, {0xF4, 0xDE}, {0xD7, 0xDE}, {0xF4, 0xDF}, {0xD8, 0xA2},
1161da177e4SLinus Torvalds {0xD8, 0xB7}, {0xD8, 0xC1}, {0xD8, 0xD1}, {0xD8, 0xF4}, {0xD9, 0xC6},
1171da177e4SLinus Torvalds {0xD9, 0xC8}, {0xD9, 0xD1}, {0xF4, 0xE0}, {0xF4, 0xE1}, {0xF4, 0xE2},
1181da177e4SLinus Torvalds {0xF4, 0xE3}, {0xF4, 0xE4}, {0xDC, 0xD3}, {0xDD, 0xC8}, {0xDD, 0xD4},
1191da177e4SLinus Torvalds {0xDD, 0xEA}, {0xDD, 0xFA}, {0xDE, 0xA4}, {0xDE, 0xB0}, {0xF4, 0xE5},
1201da177e4SLinus Torvalds {0xDE, 0xB5}, {0xDE, 0xCB}, {0xF4, 0xE6}, {0xDF, 0xB9}, {0xF4, 0xE7},
1211da177e4SLinus Torvalds {0xDF, 0xC3}, {0xF4, 0xE8}, {0xF4, 0xE9}, {0xE0, 0xD9}, {0xF4, 0xEA},
1221da177e4SLinus Torvalds {0xF4, 0xEB}, {0xE1, 0xE2}, {0xF4, 0xEC}, {0xF4, 0xED}, {0xF4, 0xEE},
1231da177e4SLinus Torvalds {0xE2, 0xC7}, {0xE3, 0xA8}, {0xE3, 0xA6}, {0xE3, 0xA9}, {0xE3, 0xAF},
1241da177e4SLinus Torvalds {0xE3, 0xB0}, {0xE3, 0xAA}, {0xE3, 0xAB}, {0xE3, 0xBC}, {0xE3, 0xC1},
1251da177e4SLinus Torvalds {0xE3, 0xBF}, {0xE3, 0xD5}, {0xE3, 0xD8}, {0xE3, 0xD6}, {0xE3, 0xDF},
1261da177e4SLinus Torvalds {0xE3, 0xE3}, {0xE3, 0xE1}, {0xE3, 0xD4}, {0xE3, 0xE9}, {0xE4, 0xA6},
1271da177e4SLinus Torvalds {0xE3, 0xF1}, {0xE3, 0xF2}, {0xE4, 0xCB}, {0xE4, 0xC1}, {0xE4, 0xC3},
1281da177e4SLinus Torvalds {0xE4, 0xBE}, {0xF4, 0xEF}, {0xE4, 0xC0}, {0xE4, 0xC7}, {0xE4, 0xBF},
1291da177e4SLinus Torvalds {0xE4, 0xE0}, {0xE4, 0xDE}, {0xE4, 0xD1}, {0xF4, 0xF0}, {0xE4, 0xDC},
1301da177e4SLinus Torvalds {0xE4, 0xD2}, {0xE4, 0xDB}, {0xE4, 0xD4}, {0xE4, 0xFA}, {0xE4, 0xEF},
1311da177e4SLinus Torvalds {0xE5, 0xB3}, {0xE5, 0xBF}, {0xE5, 0xC9}, {0xE5, 0xD0}, {0xE5, 0xE2},
1321da177e4SLinus Torvalds {0xE5, 0xEA}, {0xE5, 0xEB}, {0xF4, 0xF1}, {0xF4, 0xF2}, {0xF4, 0xF3},
1331da177e4SLinus Torvalds {0xE6, 0xE8}, {0xE6, 0xEF}, {0xE7, 0xAC}, {0xF4, 0xF4}, {0xE7, 0xAE},
1341da177e4SLinus Torvalds {0xF4, 0xF5}, {0xE7, 0xB1}, {0xF4, 0xF6}, {0xE7, 0xB2}, {0xE8, 0xB1},
1351da177e4SLinus Torvalds {0xE8, 0xB6}, {0xF4, 0xF7}, {0xF4, 0xF8}, {0xE8, 0xDD}, {0xF4, 0xF9},
1361da177e4SLinus Torvalds {0xF4, 0xFA}, {0xE9, 0xD1}, {0xF4, 0xFB}, {0xE9, 0xED}, {0xEA, 0xCD},
1371da177e4SLinus Torvalds {0xF4, 0xFC}, {0xEA, 0xDB}, {0xEA, 0xE6}, {0xEA, 0xEA}, {0xEB, 0xA5},
1381da177e4SLinus Torvalds {0xEB, 0xFB}, {0xEB, 0xFA}, {0xF4, 0xFD}, {0xEC, 0xD6}, {0xF4, 0xFE},
1391da177e4SLinus Torvalds };
1401da177e4SLinus Torvalds
1411da177e4SLinus Torvalds #define IS_EUC_IBM2JISX0208(h, l) \
1421da177e4SLinus Torvalds (((h) == 0xA2 && (l) == 0xCC) || ((h) == 0xA2 && (l) == 0xE8))
1431da177e4SLinus Torvalds
1441da177e4SLinus Torvalds /* EUC to SJIS IBM extended characters map (G3 JIS X 0212 block) */
1451da177e4SLinus Torvalds static struct {
1461da177e4SLinus Torvalds unsigned short euc;
1471da177e4SLinus Torvalds unsigned char sjis[2];
1481da177e4SLinus Torvalds } euc2sjisibm_jisx0212_map[] = {
1491da177e4SLinus Torvalds {0xA2C3, {0xFA, 0x55}}, {0xB0A9, {0xFA, 0x68}}, {0xB0C8, {0xFA, 0x69}},
1501da177e4SLinus Torvalds {0xB0D2, {0xFA, 0x6B}}, {0xB0D4, {0xFA, 0x6C}}, {0xB0E3, {0xFA, 0x6D}},
1511da177e4SLinus Torvalds {0xB0EE, {0xFA, 0x6E}}, {0xB1A3, {0xFA, 0x70}}, {0xB1A7, {0xFA, 0x6F}},
1521da177e4SLinus Torvalds {0xB1A9, {0xFA, 0x72}}, {0xB1AC, {0xFA, 0x71}}, {0xB1BB, {0xFA, 0x61}},
1531da177e4SLinus Torvalds {0xB1BE, {0xFA, 0x73}}, {0xB1C8, {0xFA, 0x76}}, {0xB1D7, {0xFA, 0x77}},
1541da177e4SLinus Torvalds {0xB1D8, {0xFA, 0x75}}, {0xB1DF, {0xFA, 0x74}}, {0xB1E1, {0xFA, 0x7A}},
1551da177e4SLinus Torvalds {0xB1E3, {0xFA, 0x78}}, {0xB1F4, {0xFA, 0x79}}, {0xB2A3, {0xFA, 0x7B}},
1561da177e4SLinus Torvalds {0xB2BB, {0xFA, 0x7D}}, {0xB2E6, {0xFA, 0x7E}}, {0xB2ED, {0xFA, 0x80}},
1571da177e4SLinus Torvalds {0xB2F5, {0xFA, 0x81}}, {0xB2FC, {0xFA, 0x82}}, {0xB3B5, {0xFA, 0x84}},
1581da177e4SLinus Torvalds {0xB3D8, {0xFA, 0x85}}, {0xB3DB, {0xFA, 0x86}}, {0xB3DD, {0xFB, 0x77}},
1591da177e4SLinus Torvalds {0xB3E5, {0xFA, 0x87}}, {0xB3EE, {0xFA, 0x88}}, {0xB3FB, {0xFA, 0x89}},
1601da177e4SLinus Torvalds {0xB4C0, {0xFA, 0x8C}}, {0xB4C7, {0xFA, 0x8D}}, {0xB4D0, {0xFA, 0x8E}},
1611da177e4SLinus Torvalds {0xB4DE, {0xFA, 0x8F}}, {0xB5AA, {0xFA, 0x91}}, {0xB5AF, {0xFA, 0x93}},
1621da177e4SLinus Torvalds {0xB5C4, {0xFA, 0x94}}, {0xB5E8, {0xFA, 0x95}}, {0xB7C2, {0xFA, 0x97}},
1631da177e4SLinus Torvalds {0xB7E4, {0xFA, 0x98}}, {0xB7E7, {0xFA, 0x9A}}, {0xB7E8, {0xFA, 0x99}},
1641da177e4SLinus Torvalds {0xB8CE, {0xFA, 0x9E}}, {0xB8E1, {0xFA, 0x9F}}, {0xB8F5, {0xFA, 0xA0}},
1651da177e4SLinus Torvalds {0xB8F7, {0xFA, 0xA1}}, {0xB8F8, {0xFA, 0xA2}}, {0xB8FC, {0xFA, 0xA3}},
1661da177e4SLinus Torvalds {0xB9AF, {0xFA, 0xA4}}, {0xB9B7, {0xFA, 0xA5}}, {0xBABE, {0xFA, 0xA6}},
1671da177e4SLinus Torvalds {0xBADB, {0xFA, 0xA7}}, {0xBAE1, {0xFA, 0xA9}}, {0xBAEB, {0xFA, 0xAB}},
1681da177e4SLinus Torvalds {0xBBB3, {0xFA, 0xAC}}, {0xBBB8, {0xFA, 0xAD}}, {0xBBCA, {0xFA, 0xAF}},
1691da177e4SLinus Torvalds {0xBBD0, {0xFA, 0xB2}}, {0xBBDE, {0xFA, 0xB3}}, {0xBBF4, {0xFA, 0xB4}},
1701da177e4SLinus Torvalds {0xBBF5, {0xFA, 0xB5}}, {0xBBF9, {0xFA, 0xB6}}, {0xBCE4, {0xFA, 0xB7}},
1711da177e4SLinus Torvalds {0xBCED, {0xFA, 0xB8}}, {0xBCF4, {0xFA, 0x67}}, {0xBCFE, {0xFA, 0xB9}},
1721da177e4SLinus Torvalds {0xBDC2, {0xFA, 0xBB}}, {0xBDE7, {0xFA, 0xBC}}, {0xBDF0, {0xFA, 0xBE}},
1731da177e4SLinus Torvalds {0xBEAC, {0xFA, 0xC0}}, {0xBEB0, {0xFA, 0xBF}}, {0xBEB3, {0xFA, 0xC2}},
1741da177e4SLinus Torvalds {0xBEBD, {0xFA, 0xC3}}, {0xBEC9, {0xFA, 0xC5}}, {0xBECD, {0xFA, 0xC4}},
1751da177e4SLinus Torvalds {0xBEE4, {0xFA, 0xC6}}, {0xBFA8, {0xFA, 0xC7}}, {0xBFC9, {0xFA, 0xC8}},
1761da177e4SLinus Torvalds {0xC0C4, {0xFA, 0xC9}}, {0xC0E4, {0xFA, 0xCA}}, {0xC0F4, {0xFA, 0xCB}},
1771da177e4SLinus Torvalds {0xC1A6, {0xFA, 0xCC}}, {0xC1F5, {0xFA, 0xCE}}, {0xC1F8, {0xFA, 0xD1}},
1781da177e4SLinus Torvalds {0xC1FC, {0xFA, 0xCF}}, {0xC2A1, {0xFA, 0xD3}}, {0xC2A5, {0xFA, 0xD4}},
1791da177e4SLinus Torvalds {0xC2AB, {0xFA, 0xD2}}, {0xC2AD, {0xFA, 0x63}}, {0xC2B8, {0xFA, 0xD6}},
1801da177e4SLinus Torvalds {0xC2BA, {0xFA, 0xD7}}, {0xC2BF, {0xFA, 0x66}}, {0xC2C4, {0xFA, 0xD9}},
1811da177e4SLinus Torvalds {0xC2D2, {0xFA, 0xDA}}, {0xC2D7, {0xFA, 0xDB}}, {0xC2DB, {0xFA, 0xDC}},
1821da177e4SLinus Torvalds {0xC2DE, {0xFA, 0xDD}}, {0xC2ED, {0xFA, 0xDE}}, {0xC2F0, {0xFA, 0xDF}},
1831da177e4SLinus Torvalds {0xC3A1, {0xFA, 0xE1}}, {0xC3B5, {0xFA, 0xE2}}, {0xC3B9, {0xFA, 0xE4}},
1841da177e4SLinus Torvalds {0xC3C9, {0xFA, 0xE3}}, {0xC3D8, {0xFA, 0xE6}}, {0xC3FC, {0xFA, 0x64}},
1851da177e4SLinus Torvalds {0xC3FE, {0xFA, 0xE7}}, {0xC4CC, {0xFA, 0xE9}}, {0xC4D9, {0xFA, 0xEB}},
1861da177e4SLinus Torvalds {0xC4EA, {0xFA, 0xEC}}, {0xC4FD, {0xFA, 0xED}}, {0xC5A7, {0xFA, 0xEF}},
1871da177e4SLinus Torvalds {0xC5B5, {0xFA, 0xF0}}, {0xC5B6, {0xFA, 0xF1}}, {0xC5D5, {0xFA, 0xF3}},
1881da177e4SLinus Torvalds {0xC6B8, {0xFA, 0xF4}}, {0xC6D7, {0xFA, 0xF5}}, {0xC6E0, {0xFA, 0xF6}},
1891da177e4SLinus Torvalds {0xC6E3, {0xFA, 0xF8}}, {0xC6EA, {0xFA, 0xF7}}, {0xC7A1, {0xFA, 0xF9}},
1901da177e4SLinus Torvalds {0xC7AB, {0xFA, 0xFA}}, {0xC7C3, {0xFA, 0xFC}}, {0xC7C7, {0xFA, 0xFB}},
1911da177e4SLinus Torvalds {0xC7CB, {0xFB, 0x40}}, {0xC7CF, {0xFB, 0x41}}, {0xC7D9, {0xFB, 0x42}},
1921da177e4SLinus Torvalds {0xC7E6, {0xFB, 0x45}}, {0xC7EB, {0xFB, 0x48}}, {0xC7EE, {0xFB, 0x46}},
1931da177e4SLinus Torvalds {0xC7F0, {0xFB, 0x49}}, {0xC7FC, {0xFB, 0x47}}, {0xC8B1, {0xFB, 0x4A}},
1941da177e4SLinus Torvalds {0xC8E5, {0xFB, 0x4B}}, {0xC8F8, {0xFB, 0x4C}}, {0xC9A6, {0xFB, 0x4D}},
1951da177e4SLinus Torvalds {0xC9AB, {0xFB, 0x4E}}, {0xC9AD, {0xFB, 0x4F}}, {0xC9CA, {0xFB, 0x51}},
1961da177e4SLinus Torvalds {0xC9D3, {0xFB, 0x52}}, {0xC9E3, {0xFB, 0x54}}, {0xC9E9, {0xFB, 0x53}},
1971da177e4SLinus Torvalds {0xC9F4, {0xFB, 0x56}}, {0xC9F5, {0xFB, 0x57}}, {0xC9FC, {0xFB, 0x55}},
1981da177e4SLinus Torvalds {0xCAB3, {0xFB, 0x59}}, {0xCABD, {0xFB, 0x5A}}, {0xCAEF, {0xFB, 0x5B}},
1991da177e4SLinus Torvalds {0xCAF1, {0xFB, 0x5C}}, {0xCBAE, {0xFB, 0x5D}}, {0xCBCA, {0xFB, 0x5F}},
2001da177e4SLinus Torvalds {0xCBE6, {0xFB, 0x60}}, {0xCBEA, {0xFB, 0x61}}, {0xCBEE, {0xFB, 0x64}},
2011da177e4SLinus Torvalds {0xCBF0, {0xFB, 0x62}}, {0xCBF4, {0xFB, 0x63}}, {0xCBF9, {0xFB, 0x66}},
2021da177e4SLinus Torvalds {0xCCA5, {0xFB, 0x65}}, {0xCCAB, {0xFB, 0x67}}, {0xCCAD, {0xFB, 0x69}},
2031da177e4SLinus Torvalds {0xCCAE, {0xFB, 0x68}}, {0xCCB2, {0xFB, 0x6A}}, {0xCCC2, {0xFB, 0x6B}},
2041da177e4SLinus Torvalds {0xCCD0, {0xFB, 0x6C}}, {0xCCD9, {0xFB, 0x6D}}, {0xCDAA, {0xFA, 0xA8}},
2051da177e4SLinus Torvalds {0xCDBB, {0xFB, 0x6F}}, {0xCEBA, {0xFB, 0x73}}, {0xCEBB, {0xFB, 0x71}},
2061da177e4SLinus Torvalds {0xCEC3, {0xFB, 0x74}}, {0xCEF2, {0xFB, 0x76}}, {0xCFD5, {0xFB, 0x78}},
2071da177e4SLinus Torvalds {0xCFE2, {0xFB, 0x79}}, {0xCFE9, {0xFB, 0x7A}}, {0xCFED, {0xFB, 0x7B}},
2081da177e4SLinus Torvalds {0xD0E5, {0xFB, 0x81}}, {0xD0E9, {0xFB, 0x83}}, {0xD1E8, {0xFB, 0x84}},
2091da177e4SLinus Torvalds {0xD1EC, {0xFB, 0x87}}, {0xD2BB, {0xFB, 0x88}}, {0xD3E1, {0xFB, 0x8A}},
2101da177e4SLinus Torvalds {0xD3E8, {0xFB, 0x8B}}, {0xD4A7, {0xFB, 0x8C}}, {0xD4D4, {0xFB, 0x8F}},
2111da177e4SLinus Torvalds {0xD4E3, {0xFA, 0x5C}}, {0xD4F2, {0xFB, 0x90}}, {0xD5AE, {0xFB, 0x91}},
2121da177e4SLinus Torvalds {0xD7DE, {0xFB, 0x93}}, {0xD8A2, {0xFB, 0x95}}, {0xD8B7, {0xFB, 0x96}},
2131da177e4SLinus Torvalds {0xD8C1, {0xFB, 0x97}}, {0xD8D1, {0xFB, 0x98}}, {0xD8F4, {0xFB, 0x99}},
2141da177e4SLinus Torvalds {0xD9A1, {0xFA, 0x60}}, {0xD9C6, {0xFB, 0x9A}}, {0xD9C8, {0xFB, 0x9B}},
2151da177e4SLinus Torvalds {0xD9D1, {0xFB, 0x9C}}, {0xDCD3, {0xFB, 0xA2}}, {0xDCDF, {0xFA, 0x5D}},
2161da177e4SLinus Torvalds {0xDDC8, {0xFB, 0xA3}}, {0xDDD4, {0xFB, 0xA4}}, {0xDDEA, {0xFB, 0xA5}},
2171da177e4SLinus Torvalds {0xDDFA, {0xFB, 0xA6}}, {0xDEA4, {0xFB, 0xA7}}, {0xDEB0, {0xFB, 0xA8}},
2181da177e4SLinus Torvalds {0xDEB5, {0xFB, 0xAA}}, {0xDECB, {0xFB, 0xAB}}, {0xDFB9, {0xFB, 0xAD}},
2191da177e4SLinus Torvalds {0xDFC3, {0xFB, 0xAF}}, {0xE0D9, {0xFB, 0xB2}}, {0xE1E2, {0xFB, 0xB5}},
2201da177e4SLinus Torvalds {0xE2C7, {0xFB, 0xB9}}, {0xE3A6, {0xFB, 0xBB}}, {0xE3A8, {0xFB, 0xBA}},
2211da177e4SLinus Torvalds {0xE3A9, {0xFB, 0xBC}}, {0xE3AA, {0xFB, 0xBF}}, {0xE3AB, {0xFB, 0xC0}},
2221da177e4SLinus Torvalds {0xE3AF, {0xFB, 0xBD}}, {0xE3B0, {0xFB, 0xBE}}, {0xE3BC, {0xFB, 0xC1}},
2231da177e4SLinus Torvalds {0xE3BF, {0xFB, 0xC3}}, {0xE3C1, {0xFB, 0xC2}}, {0xE3D4, {0xFB, 0xCA}},
2241da177e4SLinus Torvalds {0xE3D5, {0xFB, 0xC4}}, {0xE3D6, {0xFB, 0xC6}}, {0xE3D8, {0xFB, 0xC5}},
2251da177e4SLinus Torvalds {0xE3DF, {0xFB, 0xC7}}, {0xE3E1, {0xFB, 0xC9}}, {0xE3E3, {0xFB, 0xC8}},
2261da177e4SLinus Torvalds {0xE3E9, {0xFB, 0xCB}}, {0xE3F1, {0xFB, 0xCD}}, {0xE3F2, {0xFB, 0xCE}},
2271da177e4SLinus Torvalds {0xE3F8, {0xFA, 0x5F}}, {0xE4A6, {0xFB, 0xCC}}, {0xE4BE, {0xFB, 0xD2}},
2281da177e4SLinus Torvalds {0xE4BF, {0xFB, 0xD6}}, {0xE4C0, {0xFB, 0xD4}}, {0xE4C1, {0xFB, 0xD0}},
2291da177e4SLinus Torvalds {0xE4C3, {0xFB, 0xD1}}, {0xE4C7, {0xFB, 0xD5}}, {0xE4CB, {0xFB, 0xCF}},
2301da177e4SLinus Torvalds {0xE4D0, {0xFA, 0x65}}, {0xE4D1, {0xFB, 0xD9}}, {0xE4D2, {0xFB, 0xDC}},
2311da177e4SLinus Torvalds {0xE4D4, {0xFB, 0xDE}}, {0xE4DB, {0xFB, 0xDD}}, {0xE4DC, {0xFB, 0xDB}},
2321da177e4SLinus Torvalds {0xE4DE, {0xFB, 0xD8}}, {0xE4E0, {0xFB, 0xD7}}, {0xE4E9, {0xFA, 0x5E}},
2331da177e4SLinus Torvalds {0xE4EF, {0xFB, 0xE0}}, {0xE4FA, {0xFB, 0xDF}}, {0xE5B3, {0xFB, 0xE1}},
2341da177e4SLinus Torvalds {0xE5BF, {0xFB, 0xE2}}, {0xE5C9, {0xFB, 0xE3}}, {0xE5D0, {0xFB, 0xE4}},
2351da177e4SLinus Torvalds {0xE5E2, {0xFB, 0xE5}}, {0xE5EA, {0xFB, 0xE6}}, {0xE5EB, {0xFB, 0xE7}},
2361da177e4SLinus Torvalds {0xE6E8, {0xFB, 0xEB}}, {0xE6EF, {0xFB, 0xEC}}, {0xE7AC, {0xFB, 0xED}},
2371da177e4SLinus Torvalds {0xE7AE, {0xFB, 0xEF}}, {0xE7B1, {0xFB, 0xF1}}, {0xE7B2, {0xFB, 0xF3}},
2381da177e4SLinus Torvalds {0xE8B1, {0xFB, 0xF4}}, {0xE8B6, {0xFB, 0xF5}}, {0xE8DD, {0xFB, 0xF8}},
2391da177e4SLinus Torvalds {0xE9D1, {0xFB, 0xFB}}, {0xE9ED, {0xFC, 0x40}}, {0xEACD, {0xFC, 0x41}},
2401da177e4SLinus Torvalds {0xEADB, {0xFC, 0x43}}, {0xEAE6, {0xFC, 0x44}}, {0xEAEA, {0xFC, 0x45}},
2411da177e4SLinus Torvalds {0xEBA5, {0xFC, 0x46}}, {0xEBFA, {0xFC, 0x48}}, {0xEBFB, {0xFC, 0x47}},
2421da177e4SLinus Torvalds {0xECD6, {0xFC, 0x4A}},
2431da177e4SLinus Torvalds };
2441da177e4SLinus Torvalds
2451da177e4SLinus Torvalds /* EUC to SJIS IBM extended characters map (G3 Upper block) */
246*b9ec0339SDenys Vlasenko static const unsigned char euc2sjisibm_g3upper_map[][2] = {
2471da177e4SLinus Torvalds {0xFA, 0x40}, {0xFA, 0x41}, {0xFA, 0x42}, {0xFA, 0x43}, {0xFA, 0x44},
2481da177e4SLinus Torvalds {0xFA, 0x45}, {0xFA, 0x46}, {0xFA, 0x47}, {0xFA, 0x48}, {0xFA, 0x49},
2491da177e4SLinus Torvalds {0xFA, 0x4A}, {0xFA, 0x4B}, {0xFA, 0x4C}, {0xFA, 0x4D}, {0xFA, 0x4E},
2501da177e4SLinus Torvalds {0xFA, 0x4F}, {0xFA, 0x50}, {0xFA, 0x51}, {0xFA, 0x52}, {0xFA, 0x53},
2511da177e4SLinus Torvalds {0xFA, 0x56}, {0xFA, 0x57}, {0xFA, 0x58}, {0xFA, 0x59}, {0xFA, 0x5A},
2521da177e4SLinus Torvalds {0xFA, 0x62}, {0xFA, 0x6A}, {0xFA, 0x7C}, {0xFA, 0x83}, {0xFA, 0x8A},
2531da177e4SLinus Torvalds {0xFA, 0x8B}, {0xFA, 0x90}, {0xFA, 0x92}, {0xFA, 0x96}, {0xFA, 0x9B},
2541da177e4SLinus Torvalds {0xFA, 0x9C}, {0xFA, 0x9D}, {0xFA, 0xAA}, {0xFA, 0xAE}, {0xFA, 0xB0},
2551da177e4SLinus Torvalds {0xFA, 0xB1}, {0xFA, 0xBA}, {0xFA, 0xBD}, {0xFA, 0xC1}, {0xFA, 0xCD},
2561da177e4SLinus Torvalds {0xFA, 0xD0}, {0xFA, 0xD5}, {0xFA, 0xD8}, {0xFA, 0xE0}, {0xFA, 0xE5},
2571da177e4SLinus Torvalds {0xFA, 0xE8}, {0xFA, 0xEA}, {0xFA, 0xEE}, {0xFA, 0xF2}, {0xFB, 0x43},
2581da177e4SLinus Torvalds {0xFB, 0x44}, {0xFB, 0x50}, {0xFB, 0x58}, {0xFB, 0x5E}, {0xFB, 0x6E},
2591da177e4SLinus Torvalds {0xFB, 0x70}, {0xFB, 0x72}, {0xFB, 0x75}, {0xFB, 0x7C}, {0xFB, 0x7D},
2601da177e4SLinus Torvalds {0xFB, 0x7E}, {0xFB, 0x80}, {0xFB, 0x82}, {0xFB, 0x85}, {0xFB, 0x86},
2611da177e4SLinus Torvalds {0xFB, 0x89}, {0xFB, 0x8D}, {0xFB, 0x8E}, {0xFB, 0x92}, {0xFB, 0x94},
2621da177e4SLinus Torvalds {0xFB, 0x9D}, {0xFB, 0x9E}, {0xFB, 0x9F}, {0xFB, 0xA0}, {0xFB, 0xA1},
2631da177e4SLinus Torvalds {0xFB, 0xA9}, {0xFB, 0xAC}, {0xFB, 0xAE}, {0xFB, 0xB0}, {0xFB, 0xB1},
2641da177e4SLinus Torvalds {0xFB, 0xB3}, {0xFB, 0xB4}, {0xFB, 0xB6}, {0xFB, 0xB7}, {0xFB, 0xB8},
2651da177e4SLinus Torvalds {0xFB, 0xD3}, {0xFB, 0xDA}, {0xFB, 0xE8}, {0xFB, 0xE9}, {0xFB, 0xEA},
2661da177e4SLinus Torvalds {0xFB, 0xEE}, {0xFB, 0xF0}, {0xFB, 0xF2}, {0xFB, 0xF6}, {0xFB, 0xF7},
2671da177e4SLinus Torvalds {0xFB, 0xF9}, {0xFB, 0xFA}, {0xFB, 0xFC}, {0xFC, 0x42}, {0xFC, 0x49},
2681da177e4SLinus Torvalds {0xFC, 0x4B},
2691da177e4SLinus Torvalds };
2701da177e4SLinus Torvalds
2711da177e4SLinus Torvalds static inline int sjisibm2euc(unsigned char *euc, const unsigned char sjis_hi,
2721da177e4SLinus Torvalds const unsigned char sjis_lo);
2731da177e4SLinus Torvalds static inline int euc2sjisibm_jisx0212(unsigned char *sjis, const unsigned char euc_hi,
2741da177e4SLinus Torvalds const unsigned char euc_lo);
2751da177e4SLinus Torvalds static inline int euc2sjisibm_g3upper(unsigned char *sjis, const unsigned char euc_hi,
2761da177e4SLinus Torvalds const unsigned char euc_lo);
2771da177e4SLinus Torvalds static inline int euc2sjisibm(unsigned char *sjis, const unsigned char euc_hi,
2781da177e4SLinus Torvalds const unsigned char euc_lo);
2791da177e4SLinus Torvalds static inline int sjisnec2sjisibm(unsigned char *sjisibm,
2801da177e4SLinus Torvalds const unsigned char sjisnec_hi,
2811da177e4SLinus Torvalds const unsigned char sjisnec_lo);
2821da177e4SLinus Torvalds
2831da177e4SLinus Torvalds /* SJIS IBM extended characters to EUC */
sjisibm2euc(unsigned char * euc,const unsigned char sjis_hi,const unsigned char sjis_lo)2841da177e4SLinus Torvalds static inline int sjisibm2euc(unsigned char *euc, const unsigned char sjis_hi,
2851da177e4SLinus Torvalds const unsigned char sjis_lo)
2861da177e4SLinus Torvalds {
2871da177e4SLinus Torvalds int index;
2881da177e4SLinus Torvalds
2891da177e4SLinus Torvalds index = ((sjis_hi - 0xFA) * (0xFD - 0x40)) + (sjis_lo - 0x40);
2901da177e4SLinus Torvalds if (IS_EUC_IBM2JISX0208(sjisibm2euc_map[index][0],
2911da177e4SLinus Torvalds sjisibm2euc_map[index][1])) {
2921da177e4SLinus Torvalds euc[0] = sjisibm2euc_map[index][0];
2931da177e4SLinus Torvalds euc[1] = sjisibm2euc_map[index][1];
2941da177e4SLinus Torvalds return 2;
2951da177e4SLinus Torvalds } else {
2961da177e4SLinus Torvalds euc[0] = SS3;
2971da177e4SLinus Torvalds euc[1] = sjisibm2euc_map[index][0];
2981da177e4SLinus Torvalds euc[2] = sjisibm2euc_map[index][1];
2991da177e4SLinus Torvalds return 3;
3001da177e4SLinus Torvalds }
3011da177e4SLinus Torvalds }
3021da177e4SLinus Torvalds
3031da177e4SLinus Torvalds /* EUC to SJIS IBM extended characters (G3 JIS X 0212 block) */
euc2sjisibm_jisx0212(unsigned char * sjis,const unsigned char euc_hi,const unsigned char euc_lo)3041da177e4SLinus Torvalds static inline int euc2sjisibm_jisx0212(unsigned char *sjis, const unsigned char euc_hi,
3051da177e4SLinus Torvalds const unsigned char euc_lo)
3061da177e4SLinus Torvalds {
3071da177e4SLinus Torvalds int index, min_index, max_index;
3081da177e4SLinus Torvalds unsigned short euc;
3091da177e4SLinus Torvalds
3101da177e4SLinus Torvalds min_index = 0;
311e8c96f8cSTobias Klauser max_index = ARRAY_SIZE(euc2sjisibm_jisx0212_map) - 1;
3121da177e4SLinus Torvalds euc = (euc_hi << 8) | euc_lo;
3131da177e4SLinus Torvalds
3141da177e4SLinus Torvalds while (min_index <= max_index) {
3151da177e4SLinus Torvalds index = (min_index + max_index) / 2;
3161da177e4SLinus Torvalds if (euc < euc2sjisibm_jisx0212_map[index].euc)
3171da177e4SLinus Torvalds max_index = index - 1;
3181da177e4SLinus Torvalds else
3191da177e4SLinus Torvalds min_index = index + 1;
3201da177e4SLinus Torvalds if (euc == euc2sjisibm_jisx0212_map[index].euc) {
3211da177e4SLinus Torvalds sjis[0] = euc2sjisibm_jisx0212_map[index].sjis[0];
3221da177e4SLinus Torvalds sjis[1] = euc2sjisibm_jisx0212_map[index].sjis[1];
3231da177e4SLinus Torvalds return 3;
3241da177e4SLinus Torvalds }
3251da177e4SLinus Torvalds }
3261da177e4SLinus Torvalds return 0;
3271da177e4SLinus Torvalds }
3281da177e4SLinus Torvalds
3291da177e4SLinus Torvalds /* EUC to SJIS IBM extended characters (G3 Upper block) */
euc2sjisibm_g3upper(unsigned char * sjis,const unsigned char euc_hi,const unsigned char euc_lo)3301da177e4SLinus Torvalds static inline int euc2sjisibm_g3upper(unsigned char *sjis, const unsigned char euc_hi,
3311da177e4SLinus Torvalds const unsigned char euc_lo)
3321da177e4SLinus Torvalds {
3331da177e4SLinus Torvalds int index;
3341da177e4SLinus Torvalds
3351da177e4SLinus Torvalds if (euc_hi == 0xF3)
3361da177e4SLinus Torvalds index = ((euc_hi << 8) | euc_lo) - 0xF3F3;
3371da177e4SLinus Torvalds else
3381da177e4SLinus Torvalds index = ((euc_hi << 8) | euc_lo) - 0xF4A1 + 12;
3391da177e4SLinus Torvalds
340e8c96f8cSTobias Klauser if ((index < 0) || (index >= ARRAY_SIZE(euc2sjisibm_g3upper_map)))
3411da177e4SLinus Torvalds return 0;
3421da177e4SLinus Torvalds
3431da177e4SLinus Torvalds sjis[0] = euc2sjisibm_g3upper_map[index][0];
3441da177e4SLinus Torvalds sjis[1] = euc2sjisibm_g3upper_map[index][1];
3451da177e4SLinus Torvalds
3461da177e4SLinus Torvalds return 3;
3471da177e4SLinus Torvalds }
3481da177e4SLinus Torvalds
3491da177e4SLinus Torvalds /* EUC to SJIS IBM extended characters (G3 block) */
euc2sjisibm(unsigned char * sjis,const unsigned char euc_hi,const unsigned char euc_lo)3501da177e4SLinus Torvalds static inline int euc2sjisibm(unsigned char *sjis, const unsigned char euc_hi,
3511da177e4SLinus Torvalds const unsigned char euc_lo)
3521da177e4SLinus Torvalds {
3531da177e4SLinus Torvalds int n;
3541da177e4SLinus Torvalds
3551da177e4SLinus Torvalds #if 0
3561da177e4SLinus Torvalds if ((euc_hi == 0xA2) && (euc_lo == 0xCC)) {
3571da177e4SLinus Torvalds sjis[0] = 0xFA;
3581da177e4SLinus Torvalds sjis[1] = 0x54;
3591da177e4SLinus Torvalds return 2;
3601da177e4SLinus Torvalds } else if ((euc_hi == 0xA2) && (euc_lo == 0xE8)) {
3611da177e4SLinus Torvalds sjis[0] = 0xFA;
3621da177e4SLinus Torvalds sjis[1] = 0x5B;
3631da177e4SLinus Torvalds return 2;
3641da177e4SLinus Torvalds }
3651da177e4SLinus Torvalds #endif
3661da177e4SLinus Torvalds if ((n = euc2sjisibm_g3upper(sjis, euc_hi, euc_lo))) {
3671da177e4SLinus Torvalds return n;
3681da177e4SLinus Torvalds } else if ((n = euc2sjisibm_jisx0212(sjis, euc_hi, euc_lo))) {
3691da177e4SLinus Torvalds return n;
3701da177e4SLinus Torvalds }
3711da177e4SLinus Torvalds
3721da177e4SLinus Torvalds return 0;
3731da177e4SLinus Torvalds }
3741da177e4SLinus Torvalds
3751da177e4SLinus Torvalds /* NEC/IBM extended characters to IBM extended characters */
sjisnec2sjisibm(unsigned char * sjisibm,const unsigned char sjisnec_hi,const unsigned char sjisnec_lo)3761da177e4SLinus Torvalds static inline int sjisnec2sjisibm(unsigned char *sjisibm,
3771da177e4SLinus Torvalds const unsigned char sjisnec_hi,
3781da177e4SLinus Torvalds const unsigned char sjisnec_lo)
3791da177e4SLinus Torvalds {
3801da177e4SLinus Torvalds int count;
3811da177e4SLinus Torvalds
3821da177e4SLinus Torvalds if (! IS_SJIS_NECIBM(sjisnec_hi, sjisnec_lo))
3831da177e4SLinus Torvalds return 0;
3841da177e4SLinus Torvalds
3851da177e4SLinus Torvalds if ((sjisnec_hi == 0xEE) && (sjisnec_lo == 0xF9)) {
3861da177e4SLinus Torvalds sjisibm[0] = 0x81;
3871da177e4SLinus Torvalds sjisibm[1] = 0xCA;
3881da177e4SLinus Torvalds return 2;
3891da177e4SLinus Torvalds }
3901da177e4SLinus Torvalds
3911da177e4SLinus Torvalds if ((sjisnec_hi == 0xEE) && (sjisnec_lo >= 0xEF)) {
3921da177e4SLinus Torvalds count = (sjisnec_hi << 8 | sjisnec_lo)
3931da177e4SLinus Torvalds - (sjisnec_lo <= 0xF9 ? 0xEEEF : (0xEEEF - 10));
3941da177e4SLinus Torvalds } else {
3951da177e4SLinus Torvalds count = (sjisnec_hi - 0xED) * (0xFC - 0x40)
3961da177e4SLinus Torvalds + (sjisnec_lo - 0x40) + (0x5C - 0x40);
3971da177e4SLinus Torvalds if (sjisnec_lo >= 0x7F)
3981da177e4SLinus Torvalds count--;
3991da177e4SLinus Torvalds }
4001da177e4SLinus Torvalds
4011da177e4SLinus Torvalds sjisibm[0] = 0xFA + (count / (0xFC - 0x40));
4021da177e4SLinus Torvalds sjisibm[1] = 0x40 + (count % (0xFC - 0x40));
4031da177e4SLinus Torvalds if (sjisibm[1] >= 0x7F)
4041da177e4SLinus Torvalds sjisibm[1]++;
4051da177e4SLinus Torvalds
4061da177e4SLinus Torvalds return 2;
4071da177e4SLinus Torvalds }
4081da177e4SLinus Torvalds
uni2char(const wchar_t uni,unsigned char * out,int boundlen)4091da177e4SLinus Torvalds static int uni2char(const wchar_t uni,
4101da177e4SLinus Torvalds unsigned char *out, int boundlen)
4111da177e4SLinus Torvalds {
4121da177e4SLinus Torvalds int n;
4131da177e4SLinus Torvalds
4141da177e4SLinus Torvalds if (!p_nls)
4151da177e4SLinus Torvalds return -EINVAL;
4161da177e4SLinus Torvalds if ((n = p_nls->uni2char(uni, out, boundlen)) < 0)
4171da177e4SLinus Torvalds return n;
4181da177e4SLinus Torvalds
4191da177e4SLinus Torvalds /* translate SJIS into EUC-JP */
4201da177e4SLinus Torvalds if (n == 1) {
4211da177e4SLinus Torvalds if (IS_SJIS_JISX0201KANA(out[0])) {
4221da177e4SLinus Torvalds /* JIS X 0201 KANA */
4231da177e4SLinus Torvalds if (boundlen < 2)
4241da177e4SLinus Torvalds return -ENAMETOOLONG;
4251da177e4SLinus Torvalds
4261da177e4SLinus Torvalds out[1] = out[0];
4271da177e4SLinus Torvalds out[0] = SS2;
4281da177e4SLinus Torvalds return 2;
4291da177e4SLinus Torvalds }
4301da177e4SLinus Torvalds } else if (n == 2) {
4311da177e4SLinus Torvalds /* NEC/IBM extended characters to IBM extended characters */
4321da177e4SLinus Torvalds sjisnec2sjisibm(out, out[0], out[1]);
4331da177e4SLinus Torvalds
4341da177e4SLinus Torvalds if (IS_SJIS_UDC_LOW(out[0], out[1])) {
4351da177e4SLinus Torvalds /* User defined characters half low */
4361da177e4SLinus Torvalds MAP_SJIS2EUC(out[0], out[1], 0xF0, out[0], out[1], 0xF5);
4371da177e4SLinus Torvalds } else if (IS_SJIS_UDC_HI(out[0], out[1])) {
4381da177e4SLinus Torvalds /* User defined characters half high */
4391da177e4SLinus Torvalds unsigned char ch, cl;
4401da177e4SLinus Torvalds
4411da177e4SLinus Torvalds if (boundlen < 3)
4421da177e4SLinus Torvalds return -ENAMETOOLONG;
4431da177e4SLinus Torvalds
4441da177e4SLinus Torvalds n = 3; ch = out[0]; cl = out[1];
4451da177e4SLinus Torvalds out[0] = SS3;
4461da177e4SLinus Torvalds MAP_SJIS2EUC(ch, cl, 0xF5, out[1], out[2], 0xF5);
4471da177e4SLinus Torvalds } else if (IS_SJIS_IBM(out[0], out[1])) {
4481da177e4SLinus Torvalds /* IBM extended characters */
4491da177e4SLinus Torvalds unsigned char euc[3], i;
4501da177e4SLinus Torvalds
4511da177e4SLinus Torvalds n = sjisibm2euc(euc, out[0], out[1]);
4521da177e4SLinus Torvalds if (boundlen < n)
4531da177e4SLinus Torvalds return -ENAMETOOLONG;
4541da177e4SLinus Torvalds for (i = 0; i < n; i++)
4551da177e4SLinus Torvalds out[i] = euc[i];
4561da177e4SLinus Torvalds } else if (IS_SJIS_JISX0208(out[0], out[1])) {
4571da177e4SLinus Torvalds /* JIS X 0208 (include NEC special characters) */
4581da177e4SLinus Torvalds out[0] = (out[0]^0xA0)*2 + 0x5F;
4591da177e4SLinus Torvalds if (out[1] > 0x9E)
4601da177e4SLinus Torvalds out[0]++;
4611da177e4SLinus Torvalds
4621da177e4SLinus Torvalds if (out[1] < 0x7F)
4631da177e4SLinus Torvalds out[1] = out[1] + 0x61;
4641da177e4SLinus Torvalds else if (out[1] < 0x9F)
4651da177e4SLinus Torvalds out[1] = out[1] + 0x60;
4661da177e4SLinus Torvalds else
4671da177e4SLinus Torvalds out[1] = out[1] + 0x02;
4681da177e4SLinus Torvalds } else {
4691da177e4SLinus Torvalds /* Invalid characters */
4701da177e4SLinus Torvalds return -EINVAL;
4711da177e4SLinus Torvalds }
4721da177e4SLinus Torvalds }
4731da177e4SLinus Torvalds else
4741da177e4SLinus Torvalds return -EINVAL;
4751da177e4SLinus Torvalds
4761da177e4SLinus Torvalds return n;
4771da177e4SLinus Torvalds }
4781da177e4SLinus Torvalds
char2uni(const unsigned char * rawstring,int boundlen,wchar_t * uni)4791da177e4SLinus Torvalds static int char2uni(const unsigned char *rawstring, int boundlen,
4801da177e4SLinus Torvalds wchar_t *uni)
4811da177e4SLinus Torvalds {
4821da177e4SLinus Torvalds unsigned char sjis_temp[2];
4831da177e4SLinus Torvalds int euc_offset, n;
4841da177e4SLinus Torvalds
4851da177e4SLinus Torvalds if ( !p_nls )
4861da177e4SLinus Torvalds return -EINVAL;
4871da177e4SLinus Torvalds if (boundlen <= 0)
4881da177e4SLinus Torvalds return -ENAMETOOLONG;
4891da177e4SLinus Torvalds
4901da177e4SLinus Torvalds /* translate EUC-JP into SJIS */
4911da177e4SLinus Torvalds if (rawstring[0] > 0x7F) {
4921da177e4SLinus Torvalds if (rawstring[0] == SS3) {
4931da177e4SLinus Torvalds if (boundlen < 3)
4941da177e4SLinus Torvalds return -EINVAL;
4951da177e4SLinus Torvalds euc_offset = 3;
4961da177e4SLinus Torvalds
4971da177e4SLinus Torvalds if (IS_EUC_UDC_HI(rawstring[1], rawstring[2])) {
4981da177e4SLinus Torvalds /* User defined characters half high */
4991da177e4SLinus Torvalds MAP_EUC2SJIS(rawstring[1], rawstring[2], 0xF5,
5001da177e4SLinus Torvalds sjis_temp[0], sjis_temp[1], 0xF5);
5011da177e4SLinus Torvalds } else if (euc2sjisibm(sjis_temp,rawstring[1],rawstring[2])) {
5021da177e4SLinus Torvalds /* IBM extended characters */
5031da177e4SLinus Torvalds } else {
5041da177e4SLinus Torvalds /* JIS X 0212 and Invalid characters*/
5051da177e4SLinus Torvalds return -EINVAL;
5061da177e4SLinus Torvalds
5071da177e4SLinus Torvalds /* 'GETA' with SJIS coding */
5081da177e4SLinus Torvalds /* sjis_temp[0] = 0x81; */
5091da177e4SLinus Torvalds /* sjis_temp[1] = 0xAC; */
5101da177e4SLinus Torvalds }
5111da177e4SLinus Torvalds } else {
5121da177e4SLinus Torvalds if (boundlen < 2)
5131da177e4SLinus Torvalds return -EINVAL;
5141da177e4SLinus Torvalds euc_offset = 2;
5151da177e4SLinus Torvalds
5161da177e4SLinus Torvalds if (IS_EUC_JISX0201KANA(rawstring[0], rawstring[1])) {
5171da177e4SLinus Torvalds /* JIS X 0201 KANA */
5181da177e4SLinus Torvalds sjis_temp[0] = rawstring[1];
5191da177e4SLinus Torvalds sjis_temp[1] = 0x00;
5201da177e4SLinus Torvalds } else if (IS_EUC_UDC_LOW(rawstring[0], rawstring[1])) {
5211da177e4SLinus Torvalds /* User defined characters half low */
5221da177e4SLinus Torvalds MAP_EUC2SJIS(rawstring[0], rawstring[1], 0xF5,
5231da177e4SLinus Torvalds sjis_temp[0], sjis_temp[1], 0xF0);
5241da177e4SLinus Torvalds } else if (IS_EUC_JISX0208(rawstring[0], rawstring[1])) {
5251da177e4SLinus Torvalds /* JIS X 0208 (include NEC spesial characters) */
5261da177e4SLinus Torvalds sjis_temp[0] = ((rawstring[0]-0x5f)/2) ^ 0xA0;
5271da177e4SLinus Torvalds if (!(rawstring[0] & 1))
5281da177e4SLinus Torvalds sjis_temp[1] = rawstring[1] - 0x02;
5291da177e4SLinus Torvalds else if (rawstring[1] < 0xE0)
5301da177e4SLinus Torvalds sjis_temp[1] = rawstring[1] - 0x61;
5311da177e4SLinus Torvalds else
5321da177e4SLinus Torvalds sjis_temp[1] = rawstring[1] - 0x60;
5331da177e4SLinus Torvalds } else {
5341da177e4SLinus Torvalds /* Invalid characters */
5351da177e4SLinus Torvalds return -EINVAL;
5361da177e4SLinus Torvalds }
5371da177e4SLinus Torvalds }
5381da177e4SLinus Torvalds } else {
5391da177e4SLinus Torvalds euc_offset = 1;
5401da177e4SLinus Torvalds
5411da177e4SLinus Torvalds /* JIS X 0201 ROMAJI */
5421da177e4SLinus Torvalds sjis_temp[0] = rawstring[0];
5431da177e4SLinus Torvalds sjis_temp[1] = 0x00;
5441da177e4SLinus Torvalds }
5451da177e4SLinus Torvalds
5461da177e4SLinus Torvalds if ( (n = p_nls->char2uni(sjis_temp, sizeof(sjis_temp), uni)) < 0)
5471da177e4SLinus Torvalds return n;
5481da177e4SLinus Torvalds
5491da177e4SLinus Torvalds return euc_offset;
5501da177e4SLinus Torvalds }
5511da177e4SLinus Torvalds
5521da177e4SLinus Torvalds static struct nls_table table = {
5531da177e4SLinus Torvalds .charset = "euc-jp",
5541da177e4SLinus Torvalds .uni2char = uni2char,
5551da177e4SLinus Torvalds .char2uni = char2uni,
5561da177e4SLinus Torvalds };
5571da177e4SLinus Torvalds
init_nls_euc_jp(void)5581da177e4SLinus Torvalds static int __init init_nls_euc_jp(void)
5591da177e4SLinus Torvalds {
5601da177e4SLinus Torvalds p_nls = load_nls("cp932");
5611da177e4SLinus Torvalds
5621da177e4SLinus Torvalds if (p_nls) {
5631da177e4SLinus Torvalds table.charset2upper = p_nls->charset2upper;
5641da177e4SLinus Torvalds table.charset2lower = p_nls->charset2lower;
5651da177e4SLinus Torvalds return register_nls(&table);
5661da177e4SLinus Torvalds }
5671da177e4SLinus Torvalds
5681da177e4SLinus Torvalds return -EINVAL;
5691da177e4SLinus Torvalds }
5701da177e4SLinus Torvalds
exit_nls_euc_jp(void)5711da177e4SLinus Torvalds static void __exit exit_nls_euc_jp(void)
5721da177e4SLinus Torvalds {
5731da177e4SLinus Torvalds unregister_nls(&table);
5741da177e4SLinus Torvalds unload_nls(p_nls);
5751da177e4SLinus Torvalds }
5761da177e4SLinus Torvalds
5771da177e4SLinus Torvalds module_init(init_nls_euc_jp)
5781da177e4SLinus Torvalds module_exit(exit_nls_euc_jp)
5791da177e4SLinus Torvalds
5801da177e4SLinus Torvalds MODULE_LICENSE("Dual BSD/GPL");
581