summaryrefslogtreecommitdiff
path: root/usr/src/lib/iconv_modules/ko/common/wansung_to_utf.c
blob: f7abaac6422c971ddb717488789496dca28dc692 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License (the "License").
 * You may not use this file except in compliance with the License.
 *
 * You can obtain a copy of the license at src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */
/*
 * Copyright (c) 1994 by Sun Microsystems, Inc.
 */


#include <errno.h>
#include "ktable.h"
#include "hangulcode.h"


extern kcode_table	euc2utf8_tbl[];


/****  _ W A N S U N G _ T O _ U T F 8  ****/

char _wansung_to_utf8(unsigned long* chosung, unsigned long* joongsung,
				unsigned long* jongsung, unsigned short wcode)
{
	register short	h, i, j, l;
	short		ci, v, cf;
	short		disp;
	long		cfbit;

	if (wcode < 0xb0a1 || wcode > 0xc8fe)
	{  /* Hanja or special symbol */
		for (l = 0, h = MAX_E2U_NUM; l < h; )
		{
			i = (l + h) / 2;
			if (euc2utf8_tbl[i].code == wcode)
				break;
			else if (euc2utf8_tbl[l].code == wcode)
			{
				i = l;
				break;
			}
			else if (euc2utf8_tbl[h].code == wcode)
			{
				i = h;
				break;
			}
			else if (euc2utf8_tbl[i].code < wcode)
				l = i + 1;
			else
				h = i - 1;
		}

		if (euc2utf8_tbl[i].code != wcode)
			return(FAILED);

		*chosung = euc2utf8_tbl[i].utf8;
		return(HANJA_OR_SYMBOL);
	}

	if ((short)(wcode & 0xFF) < 0xA1)
		return(FAILED);

	/* Hangul processing. */
	for (h = CI_CNT, l = 0; ; )
	{
		ci = (l + h) / 2;
		if (l >= h)
			break;
		if (wcode < cmp_srchtbl[ci][0])
			h = ci - 1;
		else if (wcode < cmp_srchtbl[ci + 1][0])
			break;
		else
			l = ci + 1;
	}

	for (v = 1; ; )
	{
		if (wcode < cmp_srchtbl[ci][v])
		{
			while (!cmp_srchtbl[ci][--v])
				;
			break;
		}
		else if (v == V_CNT)
			break;
		v++;
	}

	disp = wcode - cmp_srchtbl[ci][v];
	if (((short)(cmp_srchtbl[ci][v] & BYTE_MASK) + disp) > 0xfe)
		disp -= SKIP;

	for (cfbit = cmp_bitmap[ci][v], i = -1, cf = -1; i < disp; cf++)
	{
		if (cfbit & BIT_MASK)
			i++;
		cfbit >>= 1;
	}

	if (cf == -1)
		return(FAILED);

	*chosung = 0xE18480 + ci;
	*joongsung = 0xE185A1 + v;
	/**** Original meaning is like below.
	*jongsung = (cf < 2) ? 0 <-- FILL character
			     : (cf > 25) ? 0xE18780 + cf - 26
					 : 0xE186A8 + cf - 2;
	****/
	*jongsung = (cf < 2) ? 0 : ((cf > 25) ? 0xE18766 + cf : 0xE186A6 + cf);
	return(HANGUL);
}  /* end of char _wansung_to_utf8(unsigned long*, unsigned long*,
				unsigned long*, unsigned short). */