]> asedeno.scripts.mit.edu Git - PuTTY.git/blob - unix/uxucs.c
Major destabilisation, phase 1. In this phase I've moved (I think)
[PuTTY.git] / unix / uxucs.c
1 #include <stdio.h>
2 #include <stdlib.h>
3 #include <ctype.h>
4
5 #include <time.h>
6 #include "putty.h"
7 #include "terminal.h"
8 #include "misc.h"
9
10 /*
11  * Unix Unicode-handling routines.
12  * 
13  * FIXME: currently trivial stub versions assuming all codepages
14  * are ISO8859-1.
15  */
16
17 void lpage_send(int codepage, char *buf, int len, int interactive)
18 {
19     ldisc_send(buf, len, interactive);
20 }
21
22 void luni_send(wchar_t * widebuf, int len, int interactive)
23 {
24     static char *linebuffer = 0;
25     static int linesize = 0;
26     int ratio = (in_utf(term))?6:1;
27     int i;
28     char *p;
29
30     if (len * ratio > linesize) {
31         sfree(linebuffer);
32         linebuffer = smalloc(len * ratio * 2 * sizeof(wchar_t));
33         linesize = len * ratio * 2;
34     }
35
36     if (in_utf(term)) {
37         /* UTF is a simple algorithm */
38         for (p = linebuffer, i = 0; i < len; i++) {
39             wchar_t ch = widebuf[i];
40
41             if ((ch&0xF800) == 0xD800) ch = '.';
42
43             if (ch < 0x80) {
44                 *p++ = (char) (ch);
45             } else if (ch < 0x800) {
46                 *p++ = (0xC0 | (ch >> 6));
47                 *p++ = (0x80 | (ch & 0x3F));
48             } else if (ch < 0x10000) {
49                 *p++ = (0xE0 | (ch >> 12));
50                 *p++ = (0x80 | ((ch >> 6) & 0x3F));
51                 *p++ = (0x80 | (ch & 0x3F));
52             } else if (ch < 0x200000) {
53                 *p++ = (0xF0 | (ch >> 18));
54                 *p++ = (0x80 | ((ch >> 12) & 0x3F));
55                 *p++ = (0x80 | ((ch >> 6) & 0x3F));
56                 *p++ = (0x80 | (ch & 0x3F));
57             } else if (ch < 0x4000000) {
58                 *p++ = (0xF8 | (ch >> 24));
59                 *p++ = (0x80 | ((ch >> 18) & 0x3F));
60                 *p++ = (0x80 | ((ch >> 12) & 0x3F));
61                 *p++ = (0x80 | ((ch >> 6) & 0x3F));
62                 *p++ = (0x80 | (ch & 0x3F));
63             } else {
64                 *p++ = (0xFC | (ch >> 30));
65                 *p++ = (0x80 | ((ch >> 24) & 0x3F));
66                 *p++ = (0x80 | ((ch >> 18) & 0x3F));
67                 *p++ = (0x80 | ((ch >> 12) & 0x3F));
68                 *p++ = (0x80 | ((ch >> 6) & 0x3F));
69                 *p++ = (0x80 | (ch & 0x3F));
70             }
71         }
72     } else {
73         for (p = linebuffer, i = 0; i < len; i++) {
74             wchar_t ch = widebuf[i];
75             if (ch < 0x100)
76                 *p++ = (char) ch;
77             else
78                 *p++ = '.';
79         }
80     }
81     if (p > linebuffer)
82         ldisc_send(linebuffer, p - linebuffer, interactive);
83 }
84
85 int is_dbcs_leadbyte(int codepage, char byte)
86 {
87     return 0;                          /* we don't do DBCS */
88 }
89
90 int mb_to_wc(int codepage, int flags, char *mbstr, int mblen,
91              wchar_t *wcstr, int wclen)
92 {
93     int ret = 0;
94     while (mblen > 0 && wclen > 0) {
95         *wcstr++ = (unsigned char) *mbstr++;
96         mblen--, wclen--, ret++;
97     }
98     return ret;                        /* FIXME: check error codes! */
99 }
100
101 int wc_to_mb(int codepage, int flags, wchar_t *wcstr, int wclen,
102              char *mbstr, int mblen, char *defchr, int *defused)
103 {
104     int ret = 0;
105     if (defused)
106         *defused = 0;
107     while (mblen > 0 && wclen > 0) {
108         if (*wcstr >= 0x100) {
109             if (defchr)
110                 *mbstr++ = *defchr;
111             else
112                 *mbstr++ = '\xBF';
113             if (defused)
114                 *defused = 1;
115         } else
116             *mbstr++ = (unsigned char) *wcstr;
117         wcstr++;
118         mblen--, wclen--, ret++;
119     }
120     return ret;                        /* FIXME: check error codes! */
121 }
122
123 void init_ucs(void)
124 {
125     int i;
126     /* Find the line control characters. FIXME: this is not right. */
127     for (i = 0; i < 256; i++)
128         if (i < ' ' || (i >= 0x7F && i < 0xA0))
129             unitab_ctrl[i] = i;
130         else
131             unitab_ctrl[i] = 0xFF;
132
133     for (i = 0; i < 256; i++) {
134         unitab_line[i] = unitab_scoacs[i] = i;
135         unitab_xterm[i] = (i >= 0x5F && i < 0x7F) ? ((i+1) & 0x1F) : i;
136     }
137 }