10088 ldterm_do_ioctl() shouldn't check for a NULL array
1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 /* 22 * Copyright 2008 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 * Copyright (c) 2014, Joyent, Inc. All rights reserved. 25 * Copyright 2018 OmniOS Community Edition (OmniOSce) Association. 26 */ 27 28 /* Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */ 29 /* All Rights Reserved */ 30 31 /* 32 * University Copyright- Copyright (c) 1982, 1986, 1988 33 * The Regents of the University of California 34 * All Rights Reserved 35 * 36 * University Acknowledgment- Portions of this document are derived from 37 * software developed by the University of California, Berkeley, and its 38 * contributors. 39 */ 40 41 /* 42 * Standard Streams Terminal Line Discipline module. 43 */ 44 45 #include <sys/param.h> 46 #include <sys/types.h> 47 #include <sys/termio.h> 48 #include <sys/stream.h> 49 #include <sys/conf.h> 50 #include <sys/stropts.h> 51 #include <sys/strsubr.h> 52 #include <sys/strsun.h> 53 #include <sys/strtty.h> 54 #include <sys/signal.h> 55 #include <sys/file.h> 56 #include <sys/errno.h> 57 #include <sys/debug.h> 58 #include <sys/cmn_err.h> 59 #include <sys/euc.h> 60 #include <sys/eucioctl.h> 61 #include <sys/csiioctl.h> 62 #include <sys/ptms.h> 63 #include <sys/ldterm.h> 64 #include <sys/cred.h> 65 #include <sys/ddi.h> 66 #include <sys/sunddi.h> 67 #include <sys/kmem.h> 68 #include <sys/modctl.h> 69 70 /* Time limit when draining during a close(9E) invoked by exit(2) */ 71 /* Can be set to zero to emulate the old, broken behavior */ 72 int ldterm_drain_limit = 15000000; 73 74 /* 75 * Character types. 76 */ 77 #define ORDINARY 0 78 #define CONTROL 1 79 #define BACKSPACE 2 80 #define NEWLINE 3 81 #define TAB 4 82 #define VTAB 5 83 #define RETURN 6 84 85 /* 86 * The following for EUC handling: 87 */ 88 #define T_SS2 7 89 #define T_SS3 8 90 91 /* 92 * Table indicating character classes to tty driver. In particular, 93 * if the class is ORDINARY, then the character needs no special 94 * processing on output. 95 * 96 * Characters in the C1 set are all considered CONTROL; this will 97 * work with terminals that properly use the ANSI/ISO extensions, 98 * but might cause distress with terminals that put graphics in 99 * the range 0200-0237. On the other hand, characters in that 100 * range cause even greater distress to other UNIX terminal drivers.... 101 */ 102 103 static char typetab[256] = { 104 /* 000 */ CONTROL, CONTROL, CONTROL, CONTROL, 105 /* 004 */ CONTROL, CONTROL, CONTROL, CONTROL, 106 /* 010 */ BACKSPACE, TAB, NEWLINE, CONTROL, 107 /* 014 */ VTAB, RETURN, CONTROL, CONTROL, 108 /* 020 */ CONTROL, CONTROL, CONTROL, CONTROL, 109 /* 024 */ CONTROL, CONTROL, CONTROL, CONTROL, 110 /* 030 */ CONTROL, CONTROL, CONTROL, CONTROL, 111 /* 034 */ CONTROL, CONTROL, CONTROL, CONTROL, 112 /* 040 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 113 /* 044 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 114 /* 050 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 115 /* 054 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 116 /* 060 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 117 /* 064 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 118 /* 070 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 119 /* 074 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 120 /* 100 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 121 /* 104 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 122 /* 110 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 123 /* 114 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 124 /* 120 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 125 /* 124 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 126 /* 130 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 127 /* 134 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 128 /* 140 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 129 /* 144 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 130 /* 150 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 131 /* 154 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 132 /* 160 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 133 /* 164 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 134 /* 170 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 135 /* 174 */ ORDINARY, ORDINARY, ORDINARY, CONTROL, 136 /* 200 */ CONTROL, CONTROL, CONTROL, CONTROL, 137 /* 204 */ CONTROL, CONTROL, T_SS2, T_SS3, 138 /* 210 */ CONTROL, CONTROL, CONTROL, CONTROL, 139 /* 214 */ CONTROL, CONTROL, CONTROL, CONTROL, 140 /* 220 */ CONTROL, CONTROL, CONTROL, CONTROL, 141 /* 224 */ CONTROL, CONTROL, CONTROL, CONTROL, 142 /* 230 */ CONTROL, CONTROL, CONTROL, CONTROL, 143 /* 234 */ CONTROL, CONTROL, CONTROL, CONTROL, 144 /* 240 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 145 /* 244 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 146 /* 250 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 147 /* 254 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 148 /* 260 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 149 /* 264 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 150 /* 270 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 151 /* 274 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 152 /* 300 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 153 /* 304 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 154 /* 310 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 155 /* 314 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 156 /* 320 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 157 /* 324 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 158 /* 330 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 159 /* 334 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 160 /* 340 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 161 /* 344 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 162 /* 350 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 163 /* 354 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 164 /* 360 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 165 /* 364 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 166 /* 370 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 167 /* 168 * WARNING: For EUC, 0xFF must be an ordinary character. It is used with 169 * single-byte EUC in some of the "ISO Latin Alphabet" codesets, and occupies 170 * a screen position; in those ISO sets where that position isn't used, it 171 * shouldn't make any difference. 172 */ 173 /* 374 */ ORDINARY, ORDINARY, ORDINARY, ORDINARY, 174 }; 175 176 /* 177 * Translation table for output without OLCUC. All ORDINARY-class characters 178 * translate to themselves. All other characters have a zero in the table, 179 * which stops the copying. 180 */ 181 static unsigned char notrantab[256] = { 182 /* 000 */ 0, 0, 0, 0, 0, 0, 0, 0, 183 /* 010 */ 0, 0, 0, 0, 0, 0, 0, 0, 184 /* 020 */ 0, 0, 0, 0, 0, 0, 0, 0, 185 /* 030 */ 0, 0, 0, 0, 0, 0, 0, 0, 186 /* 040 */ ' ', '!', '"', '#', '$', '%', '&', '\'', 187 /* 050 */ '(', ')', '*', '+', ',', '-', '.', '/', 188 /* 060 */ '0', '1', '2', '3', '4', '5', '6', '7', 189 /* 070 */ '8', '9', ':', ';', '<', '=', '>', '?', 190 /* 100 */ '@', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 191 /* 110 */ 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 192 /* 120 */ 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 193 /* 130 */ 'X', 'Y', 'Z', '[', '\\', ']', '^', '_', 194 /* 140 */ '`', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 195 /* 150 */ 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 196 /* 160 */ 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 197 /* 170 */ 'x', 'y', 'z', '{', '|', '}', '~', 0, 198 /* 200 */ 0, 0, 0, 0, 0, 0, 0, 0, 199 /* 210 */ 0, 0, 0, 0, 0, 0, 0, 0, 200 /* 220 */ 0, 0, 0, 0, 0, 0, 0, 0, 201 /* 230 */ 0, 0, 0, 0, 0, 0, 0, 0, 202 /* 240 */ 0240, 0241, 0242, 0243, 0244, 0245, 0246, 0247, 203 /* 250 */ 0250, 0251, 0252, 0253, 0254, 0255, 0256, 0257, 204 /* 260 */ 0260, 0261, 0262, 0263, 0264, 0265, 0266, 0267, 205 /* 270 */ 0270, 0271, 0272, 0273, 0274, 0275, 0276, 0277, 206 /* 300 */ 0300, 0301, 0302, 0303, 0304, 0305, 0306, 0307, 207 /* 310 */ 0310, 0311, 0312, 0313, 0314, 0315, 0316, 0317, 208 /* 320 */ 0320, 0321, 0322, 0323, 0324, 0325, 0326, 0327, 209 /* 330 */ 0330, 0331, 0332, 0333, 0334, 0335, 0336, 0337, 210 /* 340 */ 0340, 0341, 0342, 0343, 0344, 0345, 0346, 0347, 211 /* 350 */ 0350, 0351, 0352, 0353, 0354, 0355, 0356, 0357, 212 /* 360 */ 0360, 0361, 0362, 0363, 0364, 0365, 0366, 0367, 213 /* 214 * WARNING: as for above ISO sets, \377 may be used. Translate it to 215 * itself. 216 */ 217 /* 370 */ 0370, 0371, 0372, 0373, 0374, 0375, 0376, 0377, 218 }; 219 220 /* 221 * Translation table for output with OLCUC. All ORDINARY-class characters 222 * translate to themselves, except for lower-case letters which translate 223 * to their upper-case equivalents. All other characters have a zero in 224 * the table, which stops the copying. 225 */ 226 static unsigned char lcuctab[256] = { 227 /* 000 */ 0, 0, 0, 0, 0, 0, 0, 0, 228 /* 010 */ 0, 0, 0, 0, 0, 0, 0, 0, 229 /* 020 */ 0, 0, 0, 0, 0, 0, 0, 0, 230 /* 030 */ 0, 0, 0, 0, 0, 0, 0, 0, 231 /* 040 */ ' ', '!', '"', '#', '$', '%', '&', '\'', 232 /* 050 */ '(', ')', '*', '+', ',', '-', '.', '/', 233 /* 060 */ '0', '1', '2', '3', '4', '5', '6', '7', 234 /* 070 */ '8', '9', ':', ';', '<', '=', '>', '?', 235 /* 100 */ '@', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 236 /* 110 */ 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 237 /* 120 */ 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 238 /* 130 */ 'X', 'Y', 'Z', '[', '\\', ']', '^', '_', 239 /* 140 */ '`', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 240 /* 150 */ 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 241 /* 160 */ 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 242 /* 170 */ 'X', 'Y', 'Z', '{', '|', '}', '~', 0, 243 /* 200 */ 0, 0, 0, 0, 0, 0, 0, 0, 244 /* 210 */ 0, 0, 0, 0, 0, 0, 0, 0, 245 /* 220 */ 0, 0, 0, 0, 0, 0, 0, 0, 246 /* 230 */ 0, 0, 0, 0, 0, 0, 0, 0, 247 /* 240 */ 0240, 0241, 0242, 0243, 0244, 0245, 0246, 0247, 248 /* 250 */ 0250, 0251, 0252, 0253, 0254, 0255, 0256, 0257, 249 /* 260 */ 0260, 0261, 0262, 0263, 0264, 0265, 0266, 0267, 250 /* 270 */ 0270, 0271, 0272, 0273, 0274, 0275, 0276, 0277, 251 /* 300 */ 0300, 0301, 0302, 0303, 0304, 0305, 0306, 0307, 252 /* 310 */ 0310, 0311, 0312, 0313, 0314, 0315, 0316, 0317, 253 /* 320 */ 0320, 0321, 0322, 0323, 0324, 0325, 0326, 0327, 254 /* 330 */ 0330, 0331, 0332, 0333, 0334, 0335, 0336, 0337, 255 /* 340 */ 0340, 0341, 0342, 0343, 0344, 0345, 0346, 0347, 256 /* 350 */ 0350, 0351, 0352, 0353, 0354, 0355, 0356, 0357, 257 /* 360 */ 0360, 0361, 0362, 0363, 0364, 0365, 0366, 0367, 258 /* 259 * WARNING: as for above ISO sets, \377 may be used. Translate it to 260 * itself. 261 */ 262 /* 370 */ 0370, 0371, 0372, 0373, 0374, 0375, 0376, 0377, 263 }; 264 265 /* 266 * Input mapping table -- if an entry is non-zero, and XCASE is set, 267 * when the corresponding character is typed preceded by "\" the escape 268 * sequence is replaced by the table value. Mostly used for 269 * upper-case only terminals. 270 */ 271 static char imaptab[256] = { 272 /* 000 */ 0, 0, 0, 0, 0, 0, 0, 0, 273 /* 010 */ 0, 0, 0, 0, 0, 0, 0, 0, 274 /* 020 */ 0, 0, 0, 0, 0, 0, 0, 0, 275 /* 030 */ 0, 0, 0, 0, 0, 0, 0, 0, 276 /* 040 */ 0, '|', 0, 0, 0, 0, 0, '`', 277 /* 050 */ '{', '}', 0, 0, 0, 0, 0, 0, 278 /* 060 */ 0, 0, 0, 0, 0, 0, 0, 0, 279 /* 070 */ 0, 0, 0, 0, 0, 0, 0, 0, 280 /* 100 */ 0, 0, 0, 0, 0, 0, 0, 0, 281 /* 110 */ 0, 0, 0, 0, 0, 0, 0, 0, 282 /* 120 */ 0, 0, 0, 0, 0, 0, 0, 0, 283 /* 130 */ 0, 0, 0, 0, '\\', 0, '~', 0, 284 /* 140 */ 0, 'A', 'B', 'C', 'D', 'E', 'F', 'G', 285 /* 150 */ 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 286 /* 160 */ 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 287 /* 170 */ 'X', 'Y', 'Z', 0, 0, 0, 0, 0, 288 /* 200-377 aren't mapped */ 289 }; 290 291 /* 292 * Output mapping table -- if an entry is non-zero, and XCASE is set, 293 * the corresponding character is printed as "\" followed by the table 294 * value. Mostly used for upper-case only terminals. 295 */ 296 static char omaptab[256] = { 297 /* 000 */ 0, 0, 0, 0, 0, 0, 0, 0, 298 /* 010 */ 0, 0, 0, 0, 0, 0, 0, 0, 299 /* 020 */ 0, 0, 0, 0, 0, 0, 0, 0, 300 /* 030 */ 0, 0, 0, 0, 0, 0, 0, 0, 301 /* 040 */ 0, 0, 0, 0, 0, 0, 0, 0, 302 /* 050 */ 0, 0, 0, 0, 0, 0, 0, 0, 303 /* 060 */ 0, 0, 0, 0, 0, 0, 0, 0, 304 /* 070 */ 0, 0, 0, 0, 0, 0, 0, 0, 305 /* 100 */ 0, 'A', 'B', 'C', 'D', 'E', 'F', 'G', 306 /* 110 */ 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 307 /* 120 */ 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 308 /* 130 */ 'X', 'Y', 'Z', 0, 0, 0, 0, 0, 309 /* 140 */ '\'', 0, 0, 0, 0, 0, 0, 0, 310 /* 150 */ 0, 0, 0, 0, 0, 0, 0, 0, 311 /* 160 */ 0, 0, 0, 0, 0, 0, 0, 0, 312 /* 170 */ 0, 0, 0, '(', '!', ')', '^', 0, 313 /* 200-377 aren't mapped */ 314 }; 315 316 /* 317 * Translation table for TS_MEUC output without OLCUC. All printing ASCII 318 * characters translate to themselves. All other _bytes_ have a zero in 319 * the table, which stops the copying. This and the following table exist 320 * only so we can use the existing movtuc processing with or without OLCUC. 321 * Maybe it speeds up something...because we can copy a block of characters 322 * by only looking for zeros in the table. 323 * 324 * If we took the simple expedient of DISALLOWING "olcuc" with multi-byte 325 * processing, we could rid ourselves of both these tables and save 512 bytes; 326 * seriously, it doesn't make much sense to use olcuc with multi-byte, and 327 * it will probably never be used. Consideration should be given to disallowing 328 * the combination TS_MEUC & OLCUC. 329 */ 330 static unsigned char enotrantab[256] = { 331 /* 000 */ 0, 0, 0, 0, 0, 0, 0, 0, 332 /* 010 */ 0, 0, 0, 0, 0, 0, 0, 0, 333 /* 020 */ 0, 0, 0, 0, 0, 0, 0, 0, 334 /* 030 */ 0, 0, 0, 0, 0, 0, 0, 0, 335 /* 040 */ ' ', '!', '"', '#', '$', '%', '&', '\'', 336 /* 050 */ '(', ')', '*', '+', ',', '-', '.', '/', 337 /* 060 */ '0', '1', '2', '3', '4', '5', '6', '7', 338 /* 070 */ '8', '9', ':', ';', '<', '=', '>', '?', 339 /* 100 */ '@', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 340 /* 110 */ 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 341 /* 120 */ 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 342 /* 130 */ 'X', 'Y', 'Z', '[', '\\', ']', '^', '_', 343 /* 140 */ '`', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 344 /* 150 */ 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 345 /* 160 */ 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 346 /* 170 */ 'x', 'y', 'z', '{', '|', '}', '~', 0, 347 /* 200 - 377 aren't mapped (they're stoppers). */ 348 }; 349 350 /* 351 * Translation table for TS_MEUC output with OLCUC. All printing ASCII 352 * translate to themselves, except for lower-case letters which translate 353 * to their upper-case equivalents. All other bytes have a zero in 354 * the table, which stops the copying. Useless for ISO Latin Alphabet 355 * translations, but *sigh* OLCUC is really only defined for ASCII anyway. 356 * We only have this table so we can use the existing OLCUC processing with 357 * TS_MEUC set (multi-byte mode). Nobody would ever think of actually 358 * _using_ it...would they? 359 */ 360 static unsigned char elcuctab[256] = { 361 /* 000 */ 0, 0, 0, 0, 0, 0, 0, 0, 362 /* 010 */ 0, 0, 0, 0, 0, 0, 0, 0, 363 /* 020 */ 0, 0, 0, 0, 0, 0, 0, 0, 364 /* 030 */ 0, 0, 0, 0, 0, 0, 0, 0, 365 /* 040 */ ' ', '!', '"', '#', '$', '%', '&', '\'', 366 /* 050 */ '(', ')', '*', '+', ',', '-', '.', '/', 367 /* 060 */ '0', '1', '2', '3', '4', '5', '6', '7', 368 /* 070 */ '8', '9', ':', ';', '<', '=', '>', '?', 369 /* 100 */ '@', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 370 /* 110 */ 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 371 /* 120 */ 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 372 /* 130 */ 'X', 'Y', 'Z', '[', '\\', ']', '^', '_', 373 /* 140 */ '`', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 374 /* 150 */ 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 375 /* 160 */ 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 376 /* 170 */ 'X', 'Y', 'Z', '{', '|', '}', '~', 0, 377 /* 200 - 377 aren't mapped (they're stoppers). */ 378 }; 379 380 static struct streamtab ldtrinfo; 381 382 static struct fmodsw fsw = { 383 "ldterm", 384 &ldtrinfo, 385 D_MTQPAIR | D_MP | _D_SINGLE_INSTANCE 386 }; 387 388 static struct modlstrmod modlstrmod = { 389 &mod_strmodops, "terminal line discipline", &fsw 390 }; 391 392 393 static struct modlinkage modlinkage = { 394 MODREV_1, &modlstrmod, NULL 395 }; 396 397 398 int 399 _init(void) 400 { 401 return (mod_install(&modlinkage)); 402 } 403 404 int 405 _fini(void) 406 { 407 return (mod_remove(&modlinkage)); 408 } 409 410 int 411 _info(struct modinfo *modinfop) 412 { 413 return (mod_info(&modlinkage, modinfop)); 414 } 415 416 417 static int ldtermopen(queue_t *, dev_t *, int, int, cred_t *); 418 static int ldtermclose(queue_t *, int, cred_t *); 419 static void ldtermrput(queue_t *, mblk_t *); 420 static void ldtermrsrv(queue_t *); 421 static int ldtermrmsg(queue_t *, mblk_t *); 422 static void ldtermwput(queue_t *, mblk_t *); 423 static void ldtermwsrv(queue_t *); 424 static int ldtermwmsg(queue_t *, mblk_t *); 425 static mblk_t *ldterm_docanon(unsigned char, mblk_t *, size_t, queue_t *, 426 ldtermstd_state_t *, int *); 427 static int ldterm_unget(ldtermstd_state_t *); 428 static void ldterm_trim(ldtermstd_state_t *); 429 static void ldterm_rubout(unsigned char, queue_t *, size_t, 430 ldtermstd_state_t *); 431 static int ldterm_tabcols(ldtermstd_state_t *); 432 static void ldterm_erase(queue_t *, size_t, ldtermstd_state_t *); 433 static void ldterm_werase(queue_t *, size_t, ldtermstd_state_t *); 434 static void ldterm_kill(queue_t *, size_t, ldtermstd_state_t *); 435 static void ldterm_reprint(queue_t *, size_t, ldtermstd_state_t *); 436 static mblk_t *ldterm_dononcanon(mblk_t *, mblk_t *, size_t, queue_t *, 437 ldtermstd_state_t *); 438 static int ldterm_echo(unsigned char, queue_t *, size_t, 439 ldtermstd_state_t *); 440 static void ldterm_outchar(unsigned char, queue_t *, size_t, 441 ldtermstd_state_t *); 442 static void ldterm_outstring(unsigned char *, int, queue_t *, size_t, 443 ldtermstd_state_t *tp); 444 static mblk_t *newmsg(ldtermstd_state_t *); 445 static void ldterm_msg_upstream(queue_t *, ldtermstd_state_t *); 446 static void ldterm_wenable(void *); 447 static mblk_t *ldterm_output_msg(queue_t *, mblk_t *, mblk_t **, 448 ldtermstd_state_t *, size_t, int); 449 static void ldterm_flush_output(unsigned char, queue_t *, 450 ldtermstd_state_t *); 451 static void ldterm_dosig(queue_t *, int, unsigned char, int, int); 452 static void ldterm_do_ioctl(queue_t *, mblk_t *); 453 static int chgstropts(struct termios *, ldtermstd_state_t *, queue_t *); 454 static void ldterm_ioctl_reply(queue_t *, mblk_t *); 455 static void vmin_satisfied(queue_t *, ldtermstd_state_t *, int); 456 static void vmin_settimer(queue_t *); 457 static void vmin_timed_out(void *); 458 static void ldterm_adjust_modes(ldtermstd_state_t *); 459 static void ldterm_eucwarn(ldtermstd_state_t *); 460 static void cp_eucwioc(eucioc_t *, eucioc_t *, int); 461 static int ldterm_codeset(uchar_t, uchar_t); 462 463 static void ldterm_csi_erase(queue_t *, size_t, ldtermstd_state_t *); 464 static void ldterm_csi_werase(queue_t *, size_t, ldtermstd_state_t *); 465 466 static uchar_t ldterm_utf8_width(uchar_t *, int); 467 468 /* Codeset type specific methods for EUC, PCCS, and, UTF-8 codeset types. */ 469 static int __ldterm_dispwidth_euc(uchar_t, void *, int); 470 static int __ldterm_memwidth_euc(uchar_t, void *); 471 472 static int __ldterm_dispwidth_pccs(uchar_t, void *, int); 473 static int __ldterm_memwidth_pccs(uchar_t, void *); 474 475 static int __ldterm_dispwidth_utf8(uchar_t, void *, int); 476 static int __ldterm_memwidth_utf8(uchar_t, void *); 477 478 static const ldterm_cs_methods_t cs_methods[LDTERM_CS_TYPE_MAX + 1] = { 479 { 480 NULL, 481 NULL 482 }, 483 { 484 __ldterm_dispwidth_euc, 485 __ldterm_memwidth_euc 486 }, 487 { 488 __ldterm_dispwidth_pccs, 489 __ldterm_memwidth_pccs 490 }, 491 { 492 __ldterm_dispwidth_utf8, 493 __ldterm_memwidth_utf8 494 } 495 }; 496 497 /* 498 * The default codeset is presumably C locale's ISO 646 in EUC but 499 * the data structure at below defined as the default codeset data also 500 * support any single byte (EUC) locales. 501 */ 502 static const ldterm_cs_data_t default_cs_data = { 503 LDTERM_DATA_VERSION, 504 LDTERM_CS_TYPE_EUC, 505 (uchar_t)0, 506 (uchar_t)0, 507 (char *)NULL, 508 { 509 '\0', '\0', '\0', '\0', 510 '\0', '\0', '\0', '\0', 511 '\0', '\0', '\0', '\0', 512 '\0', '\0', '\0', '\0', 513 '\0', '\0', '\0', '\0', 514 '\0', '\0', '\0', '\0', 515 '\0', '\0', '\0', '\0', 516 '\0', '\0', '\0', '\0', 517 '\0', '\0', '\0', '\0', 518 '\0', '\0', '\0', '\0' 519 } 520 }; 521 522 /* 523 * The following tables are from either u8_textprep.c or uconv.c at 524 * usr/src/common/unicode/. The tables are used to figure out corresponding 525 * UTF-8 character byte lengths and also the validity of given character bytes. 526 */ 527 extern const int8_t u8_number_of_bytes[]; 528 extern const uchar_t u8_masks_tbl[]; 529 extern const uint8_t u8_valid_min_2nd_byte[]; 530 extern const uint8_t u8_valid_max_2nd_byte[]; 531 532 /* 533 * Unicode character width definition tables from uwidth.c: 534 */ 535 extern const ldterm_unicode_data_cell_t ldterm_ucode[][16384]; 536 537 #ifdef LDDEBUG 538 int ldterm_debug = 0; 539 #define DEBUG1(a) if (ldterm_debug == 1) printf a 540 #define DEBUG2(a) if (ldterm_debug >= 2) printf a /* allocations */ 541 #define DEBUG3(a) if (ldterm_debug >= 3) printf a /* M_CTL Stuff */ 542 #define DEBUG4(a) if (ldterm_debug >= 4) printf a /* M_READ Stuff */ 543 #define DEBUG5(a) if (ldterm_debug >= 5) printf a 544 #define DEBUG6(a) if (ldterm_debug >= 6) printf a 545 #define DEBUG7(a) if (ldterm_debug >= 7) printf a 546 #else 547 #define DEBUG1(a) 548 #define DEBUG2(a) 549 #define DEBUG3(a) 550 #define DEBUG4(a) 551 #define DEBUG5(a) 552 #define DEBUG6(a) 553 #define DEBUG7(a) 554 #endif /* LDDEBUG */ 555 556 557 /* 558 * Since most of the buffering occurs either at the stream head or in 559 * the "message currently being assembled" buffer, we have a 560 * relatively small input queue, so that blockages above us get 561 * reflected fairly quickly to the module below us. We also have a 562 * small maximum packet size, since you can put a message of that 563 * size on an empty queue no matter how much bigger than the high 564 * water mark it is. 565 */ 566 static struct module_info ldtermmiinfo = { 567 0x0bad, 568 "ldterm", 569 0, 570 _TTY_BUFSIZ, 571 _TTY_BUFSIZ, 572 LOWAT 573 }; 574 575 576 static struct qinit ldtermrinit = { 577 (int (*)())ldtermrput, 578 (int (*)())ldtermrsrv, 579 ldtermopen, 580 ldtermclose, 581 NULL, 582 &ldtermmiinfo 583 }; 584 585 586 static struct module_info ldtermmoinfo = { 587 0x0bad, 588 "ldterm", 589 0, 590 INFPSZ, 591 1, 592 0 593 }; 594 595 596 static struct qinit ldtermwinit = { 597 (int (*)())ldtermwput, 598 (int (*)())ldtermwsrv, 599 ldtermopen, 600 ldtermclose, 601 NULL, 602 &ldtermmoinfo 603 }; 604 605 606 static struct streamtab ldtrinfo = { 607 &ldtermrinit, 608 &ldtermwinit, 609 NULL, 610 NULL 611 }; 612 613 /* 614 * Dummy qbufcall callback routine used by open and close. 615 * The framework will wake up qwait_sig when we return from 616 * this routine (as part of leaving the perimeters.) 617 * (The framework enters the perimeters before calling the qbufcall() callback 618 * and leaves the perimeters after the callback routine has executed. The 619 * framework performs an implicit wakeup of any thread in qwait/qwait_sig 620 * when it leaves the perimeter. See qwait(9E).) 621 */ 622 /* ARGSUSED */ 623 static void 624 dummy_callback(void *arg) 625 {} 626 627 628 static mblk_t * 629 open_ioctl(queue_t *q, uint_t cmd) 630 { 631 mblk_t *mp; 632 bufcall_id_t id; 633 int retv; 634 635 while ((mp = mkiocb(cmd)) == NULL) { 636 id = qbufcall(q, sizeof (struct iocblk), BPRI_MED, 637 dummy_callback, NULL); 638 retv = qwait_sig(q); 639 qunbufcall(q, id); 640 if (retv == 0) 641 break; 642 } 643 return (mp); 644 } 645 646 static mblk_t * 647 open_mblk(queue_t *q, size_t len) 648 { 649 mblk_t *mp; 650 bufcall_id_t id; 651 int retv; 652 653 while ((mp = allocb(len, BPRI_MED)) == NULL) { 654 id = qbufcall(q, len, BPRI_MED, dummy_callback, NULL); 655 retv = qwait_sig(q); 656 qunbufcall(q, id); 657 if (retv == 0) 658 break; 659 } 660 return (mp); 661 } 662 663 /* 664 * Line discipline open. 665 */ 666 /* ARGSUSED1 */ 667 static int 668 ldtermopen(queue_t *q, dev_t *devp, int oflag, int sflag, cred_t *crp) 669 { 670 ldtermstd_state_t *tp; 671 mblk_t *bp, *qryp; 672 int len; 673 struct stroptions *strop; 674 struct termios *termiosp; 675 queue_t *wq; 676 677 if (q->q_ptr != NULL) { 678 return (0); /* already attached */ 679 } 680 681 tp = (ldtermstd_state_t *)kmem_zalloc(sizeof (ldtermstd_state_t), 682 KM_SLEEP); 683 684 /* 685 * Get termios defaults. These are stored as 686 * a property in the "options" node. 687 */ 688 if (ddi_getlongprop(DDI_DEV_T_ANY, ddi_root_node(), DDI_PROP_NOTPROM, 689 "ttymodes", (caddr_t)&termiosp, &len) == DDI_PROP_SUCCESS && 690 len == sizeof (struct termios)) { 691 tp->t_modes = *termiosp; 692 tp->t_amodes = *termiosp; 693 kmem_free(termiosp, len); 694 } else { 695 /* 696 * Gack! Whine about it. 697 */ 698 cmn_err(CE_WARN, "ldterm: Couldn't get ttymodes property!"); 699 } 700 bzero(&tp->t_dmodes, sizeof (struct termios)); 701 702 tp->t_state = 0; 703 704 tp->t_line = 0; 705 tp->t_col = 0; 706 707 tp->t_rocount = 0; 708 tp->t_rocol = 0; 709 710 tp->t_message = NULL; 711 tp->t_endmsg = NULL; 712 tp->t_msglen = 0; 713 tp->t_rd_request = 0; 714 715 tp->t_echomp = NULL; 716 tp->t_iocid = 0; 717 tp->t_wbufcid = 0; 718 tp->t_vtid = 0; 719 720 q->q_ptr = (caddr_t)tp; 721 WR(q)->q_ptr = (caddr_t)tp; 722 /* 723 * The following for EUC and also non-EUC codesets: 724 */ 725 tp->t_codeset = tp->t_eucleft = tp->t_eucign = tp->t_scratch_len = 0; 726 bzero(&tp->eucwioc, EUCSIZE); 727 tp->eucwioc.eucw[0] = 1; /* ASCII mem & screen width */ 728 tp->eucwioc.scrw[0] = 1; 729 tp->t_maxeuc = 1; /* the max len in bytes of an EUC char */ 730 tp->t_eucp = NULL; 731 tp->t_csmethods = cs_methods[LDTERM_CS_TYPE_EUC]; 732 tp->t_csdata = default_cs_data; 733 734 /* 735 * Try to switch to UTF-8 mode by allocating buffer for multibyte 736 * chars, keep EUC if allocation fails. 737 */ 738 if ((tp->t_eucp_mp = allocb(_TTY_BUFSIZ, BPRI_HI)) != NULL) { 739 tp->t_eucp = tp->t_eucp_mp->b_rptr; 740 tp->t_state = TS_MEUC; /* Multibyte mode. */ 741 tp->t_maxeuc = 4; /* the max len in bytes of an UTF-8 char */ 742 tp->t_csdata.codeset_type = LDTERM_CS_TYPE_UTF8; 743 tp->t_csdata.csinfo_num = 4; 744 /* locale_name needs string length with terminating NUL */ 745 tp->t_csdata.locale_name = (char *)kmem_alloc(6, KM_SLEEP); 746 (void) strcpy(tp->t_csdata.locale_name, "UTF-8"); 747 tp->t_csmethods = cs_methods[LDTERM_CS_TYPE_UTF8]; 748 } 749 tp->t_eucwarn = 0; /* no bad chars seen yet */ 750 751 qprocson(q); 752 753 /* 754 * Find out if the module below us does canonicalization; if 755 * so, we won't do it ourselves. 756 */ 757 758 if ((qryp = open_ioctl(q, MC_CANONQUERY)) == NULL) 759 goto open_abort; 760 761 /* 762 * Reformulate as an M_CTL message. The actual data will 763 * be in the b_cont field. 764 */ 765 qryp->b_datap->db_type = M_CTL; 766 wq = OTHERQ(q); 767 putnext(wq, qryp); 768 769 /* allocate a TCSBRK ioctl in case we'll need it on close */ 770 if ((qryp = open_ioctl(q, TCSBRK)) == NULL) 771 goto open_abort; 772 tp->t_drainmsg = qryp; 773 if ((bp = open_mblk(q, sizeof (int))) == NULL) 774 goto open_abort; 775 qryp->b_cont = bp; 776 777 /* 778 * Find out if the underlying driver supports proper POSIX close 779 * semantics. If not, we'll have to approximate it using TCSBRK. If 780 * it does, it will respond with MC_HAS_POSIX, and we'll catch that in 781 * the ldtermrput routine. 782 * 783 * When the ldterm_drain_limit tunable is set to zero, we behave the 784 * same as old ldterm: don't send this new message, and always use 785 * TCSBRK during close. 786 */ 787 if (ldterm_drain_limit != 0) { 788 if ((qryp = open_ioctl(q, MC_POSIXQUERY)) == NULL) 789 goto open_abort; 790 qryp->b_datap->db_type = M_CTL; 791 putnext(wq, qryp); 792 } 793 794 /* prepare to clear the water marks on close */ 795 if ((bp = open_mblk(q, sizeof (struct stroptions))) == NULL) 796 goto open_abort; 797 tp->t_closeopts = bp; 798 799 /* 800 * Set the high-water and low-water marks on the stream head 801 * to values appropriate for a terminal. Also set the "vmin" 802 * and "vtime" values to 1 and 0, turn on message-nondiscard 803 * mode (as we're in ICANON mode), and turn on "old-style 804 * NODELAY" mode. 805 */ 806 if ((bp = open_mblk(q, sizeof (struct stroptions))) == NULL) 807 goto open_abort; 808 strop = (struct stroptions *)bp->b_wptr; 809 strop->so_flags = SO_READOPT|SO_HIWAT|SO_LOWAT|SO_NDELON|SO_ISTTY; 810 strop->so_readopt = RMSGN; 811 strop->so_hiwat = _TTY_BUFSIZ; 812 strop->so_lowat = LOWAT; 813 bp->b_wptr += sizeof (struct stroptions); 814 bp->b_datap->db_type = M_SETOPTS; 815 putnext(q, bp); 816 817 return (0); /* this can become a controlling TTY */ 818 819 open_abort: 820 qprocsoff(q); 821 q->q_ptr = NULL; 822 WR(q)->q_ptr = NULL; 823 freemsg(tp->t_closeopts); 824 freemsg(tp->t_drainmsg); 825 /* Dump the state structure */ 826 kmem_free(tp, sizeof (ldtermstd_state_t)); 827 return (EINTR); 828 } 829 830 struct close_timer { 831 timeout_id_t id; 832 ldtermstd_state_t *tp; 833 }; 834 835 static void 836 drain_timed_out(void *arg) 837 { 838 struct close_timer *ctp = arg; 839 840 ctp->id = 0; 841 ctp->tp->t_state &= ~TS_IOCWAIT; 842 } 843 844 /* ARGSUSED2 */ 845 static int 846 ldtermclose(queue_t *q, int cflag, cred_t *crp) 847 { 848 ldtermstd_state_t *tp = (ldtermstd_state_t *)q->q_ptr; 849 struct stroptions *strop; 850 mblk_t *bp; 851 struct close_timer cltimer; 852 853 /* 854 * If we have an outstanding vmin timeout, cancel it. 855 */ 856 tp->t_state |= TS_CLOSE; 857 if (tp->t_vtid != 0) 858 (void) quntimeout(q, tp->t_vtid); 859 tp->t_vtid = 0; 860 861 /* 862 * Cancel outstanding qbufcall request. 863 */ 864 if (tp->t_wbufcid != 0) 865 qunbufcall(q, tp->t_wbufcid); 866 867 /* 868 * Reset the high-water and low-water marks on the stream 869 * head (?), turn on byte-stream mode, and turn off 870 * "old-style NODELAY" mode. 871 */ 872 bp = tp->t_closeopts; 873 strop = (struct stroptions *)bp->b_wptr; 874 strop->so_flags = SO_READOPT|SO_NDELOFF; 875 strop->so_readopt = RNORM; 876 bp->b_wptr += sizeof (struct stroptions); 877 bp->b_datap->db_type = M_SETOPTS; 878 putnext(q, bp); 879 880 if (cflag & (FNDELAY|FNONBLOCK)) { 881 freemsg(tp->t_drainmsg); 882 } else if ((bp = tp->t_drainmsg) != NULL) { 883 struct iocblk *iocb; 884 885 /* 886 * If the driver isn't known to have POSIX close semantics, 887 * then we have to emulate this the old way. This is done by 888 * sending down TCSBRK,1 to drain the output and waiting for 889 * the reply. 890 */ 891 iocb = (struct iocblk *)bp->b_rptr; 892 iocb->ioc_count = sizeof (int); 893 *(int *)bp->b_cont->b_rptr = 1; 894 bp->b_cont->b_wptr += sizeof (int); 895 tp->t_state |= TS_IOCWAIT; 896 tp->t_iocid = iocb->ioc_id; 897 if (!putq(WR(q), bp)) 898 putnext(WR(q), bp); 899 900 /* 901 * If we're not able to receive signals at this point, then 902 * launch a timer. This timer will prevent us from waiting 903 * forever for a signal that won't arrive. 904 */ 905 cltimer.id = 0; 906 if (!ddi_can_receive_sig() && ldterm_drain_limit != 0) { 907 cltimer.tp = tp; 908 cltimer.id = qtimeout(q, drain_timed_out, &cltimer, 909 drv_usectohz(ldterm_drain_limit)); 910 } 911 912 /* 913 * Note that the read side of ldterm and the qtimeout are 914 * protected by D_MTQPAIR, so no additional locking is needed 915 * here. 916 */ 917 while (tp->t_state & TS_IOCWAIT) { 918 if (qwait_sig(q) == 0) 919 break; 920 } 921 if (cltimer.id != 0) 922 (void) quntimeout(q, cltimer.id); 923 } 924 925 /* 926 * From here to the end, the routine does not sleep and does not 927 * reference STREAMS, so it's guaranteed to run to completion. 928 */ 929 930 qprocsoff(q); 931 932 freemsg(tp->t_message); 933 freemsg(tp->t_eucp_mp); 934 935 /* Dump the state structure, then unlink it */ 936 if (tp->t_csdata.locale_name != NULL) 937 kmem_free(tp->t_csdata.locale_name, 938 strlen(tp->t_csdata.locale_name) + 1); 939 kmem_free(tp, sizeof (ldtermstd_state_t)); 940 q->q_ptr = NULL; 941 return (0); 942 } 943 944 945 /* 946 * Put procedure for input from driver end of stream (read queue). 947 */ 948 static void 949 ldtermrput(queue_t *q, mblk_t *mp) 950 { 951 ldtermstd_state_t *tp; 952 unsigned char c; 953 queue_t *wrq = WR(q); /* write queue of ldterm mod */ 954 queue_t *nextq = q->q_next; /* queue below us */ 955 mblk_t *bp; 956 struct iocblk *qryp; 957 unsigned char *readp; 958 unsigned char *writep; 959 struct termios *emodes; /* effective modes set by driver */ 960 int dbtype; 961 962 tp = (ldtermstd_state_t *)q->q_ptr; 963 /* 964 * We received our ack from the driver saying there is nothing left to 965 * shovel out, so wake up the close routine. 966 */ 967 dbtype = DB_TYPE(mp); 968 if ((dbtype == M_IOCACK || dbtype == M_IOCNAK) && 969 (tp->t_state & (TS_CLOSE|TS_IOCWAIT)) == (TS_CLOSE|TS_IOCWAIT)) { 970 struct iocblk *iocp = (struct iocblk *)mp->b_rptr; 971 972 if (iocp->ioc_id == tp->t_iocid) { 973 tp->t_state &= ~TS_IOCWAIT; 974 freemsg(mp); 975 return; 976 } 977 } 978 979 switch (dbtype) { 980 981 default: 982 (void) putq(q, mp); 983 return; 984 985 /* 986 * Send these up unmolested 987 * 988 */ 989 case M_PCSIG: 990 case M_SIG: 991 case M_IOCNAK: 992 993 putnext(q, mp); 994 return; 995 996 case M_IOCACK: 997 998 ldterm_ioctl_reply(q, mp); 999 return; 1000 1001 case M_BREAK: 1002 1003 /* 1004 * Parity errors are sent up as M_BREAKS with single 1005 * character data (formerly handled in the driver) 1006 */ 1007 if (mp->b_wptr - mp->b_rptr == 1) { 1008 /* 1009 * IGNPAR PARMRK RESULT 1010 * off off 0 1011 * off on 3 byte sequence 1012 * on either ignored 1013 */ 1014 if (!(tp->t_amodes.c_iflag & IGNPAR)) { 1015 mp->b_wptr = mp->b_rptr; 1016 if (tp->t_amodes.c_iflag & PARMRK) { 1017 unsigned char c; 1018 1019 c = *mp->b_rptr; 1020 freemsg(mp); 1021 if ((mp = allocb(3, BPRI_HI)) == NULL) { 1022 cmn_err(CE_WARN, 1023 "ldtermrput: no blocks"); 1024 return; 1025 } 1026 mp->b_datap->db_type = M_DATA; 1027 *mp->b_wptr++ = (uchar_t)'\377'; 1028 *mp->b_wptr++ = '\0'; 1029 *mp->b_wptr++ = c; 1030 putnext(q, mp); 1031 } else { 1032 mp->b_datap->db_type = M_DATA; 1033 *mp->b_wptr++ = '\0'; 1034 putnext(q, mp); 1035 } 1036 } else { 1037 freemsg(mp); 1038 } 1039 return; 1040 } 1041 /* 1042 * We look at the apparent modes here instead of the 1043 * effective modes. Effective modes cannot be used if 1044 * IGNBRK, BRINT and PARMRK have been negotiated to 1045 * be handled by the driver. Since M_BREAK should be 1046 * sent upstream only if break processing was not 1047 * already done, it should be ok to use the apparent 1048 * modes. 1049 */ 1050 1051 if (!(tp->t_amodes.c_iflag & IGNBRK)) { 1052 if (tp->t_amodes.c_iflag & BRKINT) { 1053 ldterm_dosig(q, SIGINT, '\0', M_PCSIG, FLUSHRW); 1054 freemsg(mp); 1055 } else if (tp->t_amodes.c_iflag & PARMRK) { 1056 /* 1057 * Send '\377','\0', '\0'. 1058 */ 1059 freemsg(mp); 1060 if ((mp = allocb(3, BPRI_HI)) == NULL) { 1061 cmn_err(CE_WARN, 1062 "ldtermrput: no blocks"); 1063 return; 1064 } 1065 mp->b_datap->db_type = M_DATA; 1066 *mp->b_wptr++ = (uchar_t)'\377'; 1067 *mp->b_wptr++ = '\0'; 1068 *mp->b_wptr++ = '\0'; 1069 putnext(q, mp); 1070 } else { 1071 /* 1072 * Act as if a '\0' came in. 1073 */ 1074 freemsg(mp); 1075 if ((mp = allocb(1, BPRI_HI)) == NULL) { 1076 cmn_err(CE_WARN, 1077 "ldtermrput: no blocks"); 1078 return; 1079 } 1080 mp->b_datap->db_type = M_DATA; 1081 *mp->b_wptr++ = '\0'; 1082 putnext(q, mp); 1083 } 1084 } else { 1085 freemsg(mp); 1086 } 1087 return; 1088 1089 case M_CTL: 1090 DEBUG3(("ldtermrput: M_CTL received\n")); 1091 /* 1092 * The M_CTL has been standardized to look like an 1093 * M_IOCTL message. 1094 */ 1095 1096 if ((mp->b_wptr - mp->b_rptr) != sizeof (struct iocblk)) { 1097 DEBUG3(( 1098 "Non standard M_CTL received by ldterm module\n")); 1099 /* May be for someone else; pass it on */ 1100 putnext(q, mp); 1101 return; 1102 } 1103 qryp = (struct iocblk *)mp->b_rptr; 1104 1105 switch (qryp->ioc_cmd) { 1106 1107 case MC_PART_CANON: 1108 1109 DEBUG3(("ldtermrput: M_CTL Query Reply\n")); 1110 if (!mp->b_cont) { 1111 DEBUG3(("No information in Query Message\n")); 1112 break; 1113 } 1114 if ((mp->b_cont->b_wptr - mp->b_cont->b_rptr) == 1115 sizeof (struct termios)) { 1116 DEBUG3(("ldtermrput: M_CTL GrandScheme\n")); 1117 /* elaborate turning off scheme */ 1118 emodes = (struct termios *)mp->b_cont->b_rptr; 1119 bcopy(emodes, &tp->t_dmodes, 1120 sizeof (struct termios)); 1121 ldterm_adjust_modes(tp); 1122 break; 1123 } else { 1124 DEBUG3(("Incorrect query replysize\n")); 1125 break; 1126 } 1127 1128 case MC_NO_CANON: 1129 tp->t_state |= TS_NOCANON; 1130 /* 1131 * Note: this is very nasty. It's not clear 1132 * what the right thing to do with a partial 1133 * message is; We throw it out 1134 */ 1135 if (tp->t_message != NULL) { 1136 freemsg(tp->t_message); 1137 tp->t_message = NULL; 1138 tp->t_endmsg = NULL; 1139 tp->t_msglen = 0; 1140 tp->t_rocount = 0; 1141 tp->t_rocol = 0; 1142 if (tp->t_state & TS_MEUC) { 1143 ASSERT(tp->t_eucp_mp); 1144 tp->t_eucp = tp->t_eucp_mp->b_rptr; 1145 tp->t_codeset = 0; 1146 tp->t_eucleft = 0; 1147 } 1148 } 1149 break; 1150 1151 case MC_DO_CANON: 1152 tp->t_state &= ~TS_NOCANON; 1153 break; 1154 1155 case MC_HAS_POSIX: 1156 /* no longer any reason to drain from ldterm */ 1157 if (ldterm_drain_limit != 0) { 1158 freemsg(tp->t_drainmsg); 1159 tp->t_drainmsg = NULL; 1160 } 1161 break; 1162 1163 default: 1164 DEBUG3(("Unknown M_CTL Message\n")); 1165 break; 1166 } 1167 putnext(q, mp); /* In case anyone else has to see it */ 1168 return; 1169 1170 case M_FLUSH: 1171 /* 1172 * Flush everything we haven't looked at yet. 1173 */ 1174 1175 if ((tp->t_state & TS_ISPTSTTY) && (*mp->b_rptr & FLUSHBAND)) 1176 flushband(q, *(mp->b_rptr + 1), FLUSHDATA); 1177 else 1178 flushq(q, FLUSHDATA); 1179 1180 /* 1181 * Flush everything we have looked at. 1182 */ 1183 freemsg(tp->t_message); 1184 tp->t_message = NULL; 1185 tp->t_endmsg = NULL; 1186 tp->t_msglen = 0; 1187 tp->t_rocount = 0; 1188 tp->t_rocol = 0; 1189 if (tp->t_state & TS_MEUC) { /* EUC multi-byte */ 1190 ASSERT(tp->t_eucp_mp); 1191 tp->t_eucp = tp->t_eucp_mp->b_rptr; 1192 } 1193 putnext(q, mp); /* pass it on */ 1194 1195 /* 1196 * Relieve input flow control 1197 */ 1198 if ((tp->t_modes.c_iflag & IXOFF) && 1199 (tp->t_state & TS_TBLOCK) && 1200 !(tp->t_state & TS_IFBLOCK) && q->q_count <= TTXOLO) { 1201 tp->t_state &= ~TS_TBLOCK; 1202 (void) putnextctl(wrq, M_STARTI); 1203 DEBUG1(("M_STARTI down\n")); 1204 } 1205 return; 1206 1207 case M_DATA: 1208 break; 1209 } 1210 (void) drv_setparm(SYSRAWC, msgdsize(mp)); 1211 1212 /* 1213 * Flow control: send "start input" message if blocked and 1214 * our queue is below its low water mark. 1215 */ 1216 if ((tp->t_modes.c_iflag & IXOFF) && (tp->t_state & TS_TBLOCK) && 1217 !(tp->t_state & TS_IFBLOCK) && q->q_count <= TTXOLO) { 1218 tp->t_state &= ~TS_TBLOCK; 1219 (void) putnextctl(wrq, M_STARTI); 1220 DEBUG1(("M_STARTI down\n")); 1221 } 1222 /* 1223 * If somebody below us ("intelligent" communications 1224 * board, pseudo-tty controlled by an editor) is doing 1225 * canonicalization, don't scan it for special characters. 1226 */ 1227 if (tp->t_state & TS_NOCANON) { 1228 (void) putq(q, mp); 1229 return; 1230 } 1231 bp = mp; 1232 1233 do { 1234 readp = bp->b_rptr; 1235 writep = readp; 1236 if (tp->t_modes.c_iflag & (INLCR|IGNCR|ICRNL|IUCLC|IXON) || 1237 tp->t_modes.c_lflag & (ISIG|ICANON)) { 1238 /* 1239 * We're doing some sort of non-trivial 1240 * processing of input; look at every 1241 * character. 1242 */ 1243 while (readp < bp->b_wptr) { 1244 c = *readp++; 1245 1246 if (tp->t_modes.c_iflag & ISTRIP) 1247 c &= 0177; 1248 1249 /* 1250 * First, check that this hasn't been 1251 * escaped with the "literal next" 1252 * character. 1253 */ 1254 if (tp->t_state & TS_PLNCH) { 1255 tp->t_state &= ~TS_PLNCH; 1256 tp->t_modes.c_lflag &= ~FLUSHO; 1257 *writep++ = c; 1258 continue; 1259 } 1260 /* 1261 * Setting a special character to NUL 1262 * disables it, so if this character 1263 * is NUL, it should not be compared 1264 * with any of the special characters. 1265 * It should, however, restart frozen 1266 * output if IXON and IXANY are set. 1267 */ 1268 if (c == _POSIX_VDISABLE) { 1269 if (tp->t_modes.c_iflag & IXON && 1270 tp->t_state & TS_TTSTOP && 1271 tp->t_modes.c_lflag & IEXTEN && 1272 tp->t_modes.c_iflag & IXANY) { 1273 tp->t_state &= 1274 ~(TS_TTSTOP|TS_OFBLOCK); 1275 (void) putnextctl(wrq, M_START); 1276 } 1277 tp->t_modes.c_lflag &= ~FLUSHO; 1278 *writep++ = c; 1279 continue; 1280 } 1281 /* 1282 * If stopped, start if you can; if 1283 * running, stop if you must. 1284 */ 1285 if (tp->t_modes.c_iflag & IXON) { 1286 if (tp->t_state & TS_TTSTOP) { 1287 if (c == 1288 tp->t_modes.c_cc[VSTART] || 1289 (tp->t_modes.c_lflag & 1290 IEXTEN && 1291 tp->t_modes.c_iflag & 1292 IXANY)) { 1293 tp->t_state &= 1294 ~(TS_TTSTOP | 1295 TS_OFBLOCK); 1296 (void) putnextctl(wrq, 1297 M_START); 1298 } 1299 } else { 1300 if (c == 1301 tp->t_modes.c_cc[VSTOP]) { 1302 tp->t_state |= 1303 TS_TTSTOP; 1304 (void) putnextctl(wrq, 1305 M_STOP); 1306 } 1307 } 1308 if (c == tp->t_modes.c_cc[VSTOP] || 1309 c == tp->t_modes.c_cc[VSTART]) 1310 continue; 1311 } 1312 /* 1313 * Check for "literal next" character 1314 * and "flush output" character. 1315 * Note that we omit checks for ISIG 1316 * and ICANON, since the IEXTEN 1317 * setting subsumes them. 1318 */ 1319 if (tp->t_modes.c_lflag & IEXTEN) { 1320 if (c == tp->t_modes.c_cc[VLNEXT]) { 1321 /* 1322 * Remember that we saw a 1323 * "literal next" while 1324 * scanning input, but leave 1325 * leave it in the message so 1326 * that the service routine 1327 * can see it too. 1328 */ 1329 tp->t_state |= TS_PLNCH; 1330 tp->t_modes.c_lflag &= ~FLUSHO; 1331 *writep++ = c; 1332 continue; 1333 } 1334 if (c == tp->t_modes.c_cc[VDISCARD]) { 1335 ldterm_flush_output(c, wrq, tp); 1336 continue; 1337 } 1338 } 1339 tp->t_modes.c_lflag &= ~FLUSHO; 1340 1341 /* 1342 * Check for signal-generating 1343 * characters. 1344 */ 1345 if (tp->t_modes.c_lflag & ISIG) { 1346 if (c == tp->t_modes.c_cc[VINTR]) { 1347 ldterm_dosig(q, SIGINT, c, 1348 M_PCSIG, FLUSHRW); 1349 continue; 1350 } 1351 if (c == tp->t_modes.c_cc[VQUIT]) { 1352 ldterm_dosig(q, SIGQUIT, c, 1353 M_PCSIG, FLUSHRW); 1354 continue; 1355 } 1356 if (c == tp->t_modes.c_cc[VSWTCH]) { 1357 /* 1358 * Ancient SXT support; discard 1359 * character without action. 1360 */ 1361 continue; 1362 } 1363 if (c == tp->t_modes.c_cc[VSUSP]) { 1364 ldterm_dosig(q, SIGTSTP, c, 1365 M_PCSIG, FLUSHRW); 1366 continue; 1367 } 1368 if ((tp->t_modes.c_lflag & IEXTEN) && 1369 (c == tp->t_modes.c_cc[VDSUSP])) { 1370 ldterm_dosig(q, SIGTSTP, c, 1371 M_SIG, 0); 1372 continue; 1373 } 1374 1375 /* 1376 * Consumers do not expect the ^T to be 1377 * echoed out when we generate a 1378 * VSTATUS. 1379 */ 1380 if (c == tp->t_modes.c_cc[VSTATUS]) { 1381 ldterm_dosig(q, SIGINFO, '\0', 1382 M_PCSIG, FLUSHRW); 1383 continue; 1384 } 1385 } 1386 /* 1387 * Throw away CR if IGNCR set, or 1388 * turn it into NL if ICRNL set. 1389 */ 1390 if (c == '\r') { 1391 if (tp->t_modes.c_iflag & IGNCR) 1392 continue; 1393 if (tp->t_modes.c_iflag & ICRNL) 1394 c = '\n'; 1395 } else { 1396 /* 1397 * Turn NL into CR if INLCR 1398 * set. 1399 */ 1400 if (c == '\n' && 1401 tp->t_modes.c_iflag & INLCR) 1402 c = '\r'; 1403 } 1404 1405 /* 1406 * Map upper case input to lower case 1407 * if IUCLC flag set. 1408 */ 1409 if (tp->t_modes.c_iflag & IUCLC && 1410 c >= 'A' && c <= 'Z') 1411 c += 'a' - 'A'; 1412 1413 /* 1414 * Put the possibly-transformed 1415 * character back in the message. 1416 */ 1417 *writep++ = c; 1418 } 1419 1420 /* 1421 * If we didn't copy some characters because 1422 * we were ignoring them, fix the size of the 1423 * data block by adjusting the write pointer. 1424 * XXX This may result in a zero-length 1425 * block; will this cause anybody gastric 1426 * distress? 1427 */ 1428 bp->b_wptr -= (readp - writep); 1429 } else { 1430 /* 1431 * We won't be doing anything other than 1432 * possibly stripping the input. 1433 */ 1434 if (tp->t_modes.c_iflag & ISTRIP) { 1435 while (readp < bp->b_wptr) 1436 *writep++ = *readp++ & 0177; 1437 } 1438 tp->t_modes.c_lflag &= ~FLUSHO; 1439 } 1440 1441 } while ((bp = bp->b_cont) != NULL); /* next block, if any */ 1442 1443 /* 1444 * Queue the message for service procedure if the 1445 * queue is not empty or canputnext() fails or 1446 * tp->t_state & TS_RESCAN is true. 1447 */ 1448 1449 if (q->q_first != NULL || !bcanputnext(q, mp->b_band) || 1450 (tp->t_state & TS_RESCAN)) 1451 (void) putq(q, mp); 1452 else 1453 (void) ldtermrmsg(q, mp); 1454 1455 /* 1456 * Flow control: send "stop input" message if our queue is 1457 * approaching its high-water mark. The message will be 1458 * dropped on the floor in the service procedure, if we 1459 * cannot ship it up and we have had it upto our neck! 1460 * 1461 * Set QWANTW to ensure that the read queue service procedure 1462 * gets run when nextq empties up again, so that it can 1463 * unstop the input. 1464 */ 1465 if ((tp->t_modes.c_iflag & IXOFF) && !(tp->t_state & TS_TBLOCK) && 1466 q->q_count >= TTXOHI) { 1467 mutex_enter(QLOCK(nextq)); 1468 nextq->q_flag |= QWANTW; 1469 mutex_exit(QLOCK(nextq)); 1470 tp->t_state |= TS_TBLOCK; 1471 (void) putnextctl(wrq, M_STOPI); 1472 DEBUG1(("M_STOPI down\n")); 1473 } 1474 } 1475 1476 1477 /* 1478 * Line discipline input server processing. Erase/kill and escape 1479 * ('\') processing, gathering into messages, upper/lower case input 1480 * mapping. 1481 */ 1482 static void 1483 ldtermrsrv(queue_t *q) 1484 { 1485 ldtermstd_state_t *tp; 1486 mblk_t *mp; 1487 1488 tp = (ldtermstd_state_t *)q->q_ptr; 1489 1490 if (tp->t_state & TS_RESCAN) { 1491 /* 1492 * Canonicalization was turned on or off. Put the 1493 * message being assembled back in the input queue, 1494 * so that we rescan it. 1495 */ 1496 if (tp->t_message != NULL) { 1497 DEBUG5(("RESCAN WAS SET; put back in q\n")); 1498 if (tp->t_msglen != 0) 1499 (void) putbq(q, tp->t_message); 1500 else 1501 freemsg(tp->t_message); 1502 tp->t_message = NULL; 1503 tp->t_endmsg = NULL; 1504 tp->t_msglen = 0; 1505 } 1506 if (tp->t_state & TS_MEUC) { 1507 ASSERT(tp->t_eucp_mp); 1508 tp->t_eucp = tp->t_eucp_mp->b_rptr; 1509 tp->t_codeset = 0; 1510 tp->t_eucleft = 0; 1511 } 1512 tp->t_state &= ~TS_RESCAN; 1513 } 1514 1515 while ((mp = getq(q)) != NULL) { 1516 if (!ldtermrmsg(q, mp)) 1517 break; 1518 } 1519 1520 /* 1521 * Flow control: send start message if blocked and our queue 1522 * is below its low water mark. 1523 */ 1524 if ((tp->t_modes.c_iflag & IXOFF) && (tp->t_state & TS_TBLOCK) && 1525 !(tp->t_state & TS_IFBLOCK) && q->q_count <= TTXOLO) { 1526 tp->t_state &= ~TS_TBLOCK; 1527 (void) putctl(WR(q), M_STARTI); 1528 } 1529 } 1530 1531 /* 1532 * This routine is called from both ldtermrput and ldtermrsrv to 1533 * do the actual work of dealing with mp. Return 1 on sucesss and 1534 * 0 on failure. 1535 */ 1536 static int 1537 ldtermrmsg(queue_t *q, mblk_t *mp) 1538 { 1539 unsigned char c; 1540 int dofree; 1541 int status = 1; 1542 size_t ebsize; 1543 mblk_t *bp; 1544 mblk_t *bpt; 1545 ldtermstd_state_t *tp; 1546 1547 bpt = NULL; 1548 1549 tp = (ldtermstd_state_t *)q->q_ptr; 1550 1551 if (mp->b_datap->db_type <= QPCTL && !bcanputnext(q, mp->b_band)) { 1552 /* 1553 * Stream head is flow controlled. If echo is 1554 * turned on, flush the read side or send a 1555 * bell down the line to stop input and 1556 * process the current message. 1557 * Otherwise(putbq) the user will not see any 1558 * response to to the typed input. Typically 1559 * happens if there is no reader process. 1560 * Note that you will loose the data in this 1561 * case if the data is coming too fast. There 1562 * is an assumption here that if ECHO is 1563 * turned on its some user typing the data on 1564 * a terminal and its not network. 1565 */ 1566 if (tp->t_modes.c_lflag & ECHO) { 1567 if ((tp->t_modes.c_iflag & IMAXBEL) && 1568 (tp->t_modes.c_lflag & ICANON)) { 1569 freemsg(mp); 1570 if (canputnext(WR(q))) 1571 ldterm_outchar(CTRL('g'), WR(q), 4, tp); 1572 status = 0; 1573 goto echo; 1574 } else { 1575 (void) putctl1(q, M_FLUSH, FLUSHR); 1576 } 1577 } else { 1578 (void) putbq(q, mp); 1579 status = 0; 1580 goto out; /* read side is blocked */ 1581 } 1582 } 1583 switch (mp->b_datap->db_type) { 1584 1585 default: 1586 putnext(q, mp); /* pass it on */ 1587 goto out; 1588 1589 case M_HANGUP: 1590 /* 1591 * Flush everything we haven't looked at yet. 1592 */ 1593 flushq(q, FLUSHDATA); 1594 1595 /* 1596 * Flush everything we have looked at. 1597 */ 1598 freemsg(tp->t_message); 1599 tp->t_message = NULL; 1600 tp->t_endmsg = NULL; 1601 tp->t_msglen = 0; 1602 /* 1603 * XXX should we set read request 1604 * tp->t_rd_request to NULL? 1605 */ 1606 tp->t_rocount = 0; /* if it hasn't been typed */ 1607 tp->t_rocol = 0; /* it hasn't been echoed :-) */ 1608 if (tp->t_state & TS_MEUC) { 1609 ASSERT(tp->t_eucp_mp); 1610 tp->t_eucp = tp->t_eucp_mp->b_rptr; 1611 } 1612 /* 1613 * Restart output, since it's probably got 1614 * nowhere to go anyway, and we're probably 1615 * not going to see another ^Q for a while. 1616 */ 1617 if (tp->t_state & TS_TTSTOP) { 1618 tp->t_state &= ~(TS_TTSTOP|TS_OFBLOCK); 1619 (void) putnextctl(WR(q), M_START); 1620 } 1621 /* 1622 * This message will travel up the read 1623 * queue, flushing as it goes, get turned 1624 * around at the stream head, and travel back 1625 * down the write queue, flushing as it goes. 1626 */ 1627 (void) putnextctl1(q, M_FLUSH, FLUSHW); 1628 1629 /* 1630 * This message will travel down the write 1631 * queue, flushing as it goes, get turned 1632 * around at the driver, and travel back up 1633 * the read queue, flushing as it goes. 1634 */ 1635 (void) putctl1(WR(q), M_FLUSH, FLUSHR); 1636 1637 /* 1638 * Now that that's done, we send a SIGCONT 1639 * upstream, followed by the M_HANGUP. 1640 */ 1641 /* (void) putnextctl1(q, M_PCSIG, SIGCONT); */ 1642 putnext(q, mp); 1643 goto out; 1644 1645 case M_IOCACK: 1646 1647 /* 1648 * Augment whatever information the driver is 1649 * returning with the information we supply. 1650 */ 1651 ldterm_ioctl_reply(q, mp); 1652 goto out; 1653 1654 case M_DATA: 1655 break; 1656 } 1657 1658 /* 1659 * This is an M_DATA message. 1660 */ 1661 1662 /* 1663 * If somebody below us ("intelligent" communications 1664 * board, pseudo-tty controlled by an editor) is 1665 * doing canonicalization, don't scan it for special 1666 * characters. 1667 */ 1668 if (tp->t_state & TS_NOCANON) { 1669 putnext(q, mp); 1670 goto out; 1671 } 1672 bp = mp; 1673 1674 if ((bpt = newmsg(tp)) != NULL) { 1675 mblk_t *bcont; 1676 1677 do { 1678 ASSERT(bp->b_wptr >= bp->b_rptr); 1679 ebsize = bp->b_wptr - bp->b_rptr; 1680 if (ebsize > EBSIZE) 1681 ebsize = EBSIZE; 1682 bcont = bp->b_cont; 1683 if (CANON_MODE) { 1684 /* 1685 * By default, free the message once processed 1686 */ 1687 dofree = 1; 1688 1689 /* 1690 * update sysinfo canch 1691 * character. The value of 1692 * canch may vary as compared 1693 * to character tty 1694 * implementation. 1695 */ 1696 while (bp->b_rptr < bp->b_wptr) { 1697 c = *bp->b_rptr++; 1698 if ((bpt = ldterm_docanon(c, 1699 bpt, ebsize, q, tp, &dofree)) == 1700 NULL) 1701 break; 1702 } 1703 /* 1704 * Release this block or put back on queue. 1705 */ 1706 if (dofree) 1707 freeb(bp); 1708 else { 1709 (void) putbq(q, bp); 1710 break; 1711 } 1712 } else 1713 bpt = ldterm_dononcanon(bp, bpt, ebsize, q, tp); 1714 if (bpt == NULL) { 1715 cmn_err(CE_WARN, 1716 "ldtermrsrv: out of blocks"); 1717 freemsg(bcont); 1718 break; 1719 } 1720 } while ((bp = bcont) != NULL); 1721 } 1722 echo: 1723 /* 1724 * Send whatever we echoed downstream. 1725 */ 1726 if (tp->t_echomp != NULL) { 1727 if (canputnext(WR(q))) 1728 putnext(WR(q), tp->t_echomp); 1729 else 1730 freemsg(tp->t_echomp); 1731 tp->t_echomp = NULL; 1732 } 1733 1734 out: 1735 return (status); 1736 } 1737 1738 1739 /* 1740 * Do canonical mode input; check whether this character is to be 1741 * treated as a special character - if so, check whether it's equal 1742 * to any of the special characters and handle it accordingly. 1743 * Otherwise, just add it to the current line. 1744 */ 1745 static mblk_t * 1746 ldterm_docanon(uchar_t c, mblk_t *bpt, size_t ebsize, queue_t *q, 1747 ldtermstd_state_t *tp, int *dofreep) 1748 { 1749 queue_t *wrq = WR(q); 1750 int i; 1751 1752 /* 1753 * If the previous character was the "literal next" 1754 * character, treat this character as regular input. 1755 */ 1756 if (tp->t_state & TS_SLNCH) 1757 goto escaped; 1758 1759 /* 1760 * Setting a special character to NUL disables it, so if this 1761 * character is NUL, it should not be compared with any of 1762 * the special characters. 1763 */ 1764 if (c == _POSIX_VDISABLE) { 1765 tp->t_state &= ~TS_QUOT; 1766 goto escaped; 1767 } 1768 /* 1769 * If this character is the literal next character, echo it 1770 * as '^', backspace over it, and record that fact. 1771 */ 1772 if ((tp->t_modes.c_lflag & IEXTEN) && c == tp->t_modes.c_cc[VLNEXT]) { 1773 if (tp->t_modes.c_lflag & ECHO) 1774 ldterm_outstring((unsigned char *)"^\b", 2, wrq, 1775 ebsize, tp); 1776 tp->t_state |= TS_SLNCH; 1777 goto out; 1778 } 1779 /* 1780 * Check for the editing character. If the display width of 1781 * the last byte at the canonical buffer is not one and also 1782 * smaller than or equal to UNKNOWN_WIDTH, the character at 1783 * the end of the buffer is a multi-byte and/or multi-column 1784 * character. 1785 */ 1786 if (c == tp->t_modes.c_cc[VERASE] || c == tp->t_modes.c_cc[VERASE2]) { 1787 if (tp->t_state & TS_QUOT) { 1788 /* 1789 * Get rid of the backslash, and put the 1790 * erase character in its place. 1791 */ 1792 ldterm_erase(wrq, ebsize, tp); 1793 bpt = tp->t_endmsg; 1794 goto escaped; 1795 } else { 1796 if ((tp->t_state & TS_MEUC) && tp->t_msglen && 1797 (*(tp->t_eucp - 1) != 1 && 1798 *(tp->t_eucp - 1) <= UNKNOWN_WIDTH)) 1799 ldterm_csi_erase(wrq, ebsize, tp); 1800 else 1801 ldterm_erase(wrq, ebsize, tp); 1802 bpt = tp->t_endmsg; 1803 goto out; 1804 } 1805 } 1806 if ((tp->t_modes.c_lflag & IEXTEN) && c == tp->t_modes.c_cc[VWERASE]) { 1807 /* 1808 * Do "ASCII word" or "multibyte character token/chunk" erase. 1809 */ 1810 if (tp->t_state & TS_MEUC) 1811 ldterm_csi_werase(wrq, ebsize, tp); 1812 else 1813 ldterm_werase(wrq, ebsize, tp); 1814 bpt = tp->t_endmsg; 1815 goto out; 1816 } 1817 if (c == tp->t_modes.c_cc[VKILL]) { 1818 if (tp->t_state & TS_QUOT) { 1819 /* 1820 * Get rid of the backslash, and put the kill 1821 * character in its place. 1822 */ 1823 ldterm_erase(wrq, ebsize, tp); 1824 bpt = tp->t_endmsg; 1825 goto escaped; 1826 } else { 1827 ldterm_kill(wrq, ebsize, tp); 1828 bpt = tp->t_endmsg; 1829 goto out; 1830 } 1831 } 1832 if ((tp->t_modes.c_lflag & IEXTEN) && c == tp->t_modes.c_cc[VREPRINT]) { 1833 ldterm_reprint(wrq, ebsize, tp); 1834 goto out; 1835 } 1836 /* 1837 * If the preceding character was a backslash: if the current 1838 * character is an EOF, get rid of the backslash and treat 1839 * the EOF as data; if we're in XCASE mode and the current 1840 * character is part of a backslash-X escape sequence, 1841 * process it; otherwise, just treat the current character 1842 * normally. 1843 */ 1844 if (tp->t_state & TS_QUOT) { 1845 tp->t_state &= ~TS_QUOT; 1846 if (c == tp->t_modes.c_cc[VEOF]) { 1847 /* 1848 * EOF character. Since it's escaped, get rid 1849 * of the backslash and put the EOF character 1850 * in its place. 1851 */ 1852 ldterm_erase(wrq, ebsize, tp); 1853 bpt = tp->t_endmsg; 1854 } else { 1855 /* 1856 * If we're in XCASE mode, and the current 1857 * character is part of a backslash-X 1858 * sequence, get rid of the backslash and 1859 * replace the current character with what 1860 * that sequence maps to. 1861 */ 1862 if ((tp->t_modes.c_lflag & XCASE) && 1863 imaptab[c] != '\0') { 1864 ldterm_erase(wrq, ebsize, tp); 1865 bpt = tp->t_endmsg; 1866 c = imaptab[c]; 1867 } 1868 } 1869 } else { 1870 /* 1871 * Previous character wasn't backslash; check whether 1872 * this was the EOF character. 1873 */ 1874 if (c == tp->t_modes.c_cc[VEOF]) { 1875 /* 1876 * EOF character. Don't echo it unless 1877 * ECHOCTL is set, don't stuff it in the 1878 * current line, but send the line up the 1879 * stream. 1880 */ 1881 if ((tp->t_modes.c_lflag & ECHOCTL) && 1882 (tp->t_modes.c_lflag & IEXTEN) && 1883 (tp->t_modes.c_lflag & ECHO)) { 1884 i = ldterm_echo(c, wrq, ebsize, tp); 1885 while (i > 0) { 1886 ldterm_outchar('\b', wrq, ebsize, tp); 1887 i--; 1888 } 1889 } 1890 bpt->b_datap->db_type = M_DATA; 1891 ldterm_msg_upstream(q, tp); 1892 if (!canputnext(q)) { 1893 bpt = NULL; 1894 *dofreep = 0; 1895 } else { 1896 bpt = newmsg(tp); 1897 *dofreep = 1; 1898 } 1899 goto out; 1900 } 1901 } 1902 1903 escaped: 1904 /* 1905 * First, make sure we can fit one WHOLE multi-byte char in the 1906 * buffer. This is one place where we have overhead even if 1907 * not in multi-byte mode; the overhead is subtracting 1908 * tp->t_maxeuc from MAX_CANON before checking. 1909 * 1910 * Allows MAX_CANON bytes in the buffer before throwing awaying 1911 * the the overflow of characters. 1912 */ 1913 if ((tp->t_msglen > ((_TTY_BUFSIZ + 1) - (int)tp->t_maxeuc)) && 1914 !((tp->t_state & TS_MEUC) && tp->t_eucleft)) { 1915 1916 /* 1917 * Byte will cause line to overflow, or the next EUC 1918 * won't fit: Ring the bell or discard all input, and 1919 * don't save the byte away. 1920 */ 1921 if (tp->t_modes.c_iflag & IMAXBEL) { 1922 if (canputnext(wrq)) 1923 ldterm_outchar(CTRL('g'), wrq, ebsize, tp); 1924 goto out; 1925 } else { 1926 /* 1927 * MAX_CANON processing. free everything in 1928 * the current line and start with the 1929 * current character as the first character. 1930 */ 1931 DEBUG7(("ldterm_docanon: MAX_CANON processing\n")); 1932 freemsg(tp->t_message); 1933 tp->t_message = NULL; 1934 tp->t_endmsg = NULL; 1935 tp->t_msglen = 0; 1936 tp->t_rocount = 0; /* if it hasn't been type */ 1937 tp->t_rocol = 0; /* it hasn't been echoed :-) */ 1938 if (tp->t_state & TS_MEUC) { 1939 ASSERT(tp->t_eucp_mp); 1940 tp->t_eucp = tp->t_eucp_mp->b_rptr; 1941 } 1942 tp->t_state &= ~TS_SLNCH; 1943 bpt = newmsg(tp); 1944 } 1945 } 1946 /* 1947 * Add the character to the current line. 1948 */ 1949 if (bpt->b_wptr >= bpt->b_datap->db_lim) { 1950 /* 1951 * No more room in this mblk; save this one away, and 1952 * allocate a new one. 1953 */ 1954 bpt->b_datap->db_type = M_DATA; 1955 if ((bpt = allocb(IBSIZE, BPRI_MED)) == NULL) 1956 goto out; 1957 1958 /* 1959 * Chain the new one to the end of the old one, and 1960 * mark it as the last block in the current line. 1961 */ 1962 tp->t_endmsg->b_cont = bpt; 1963 tp->t_endmsg = bpt; 1964 } 1965 *bpt->b_wptr++ = c; 1966 tp->t_msglen++; /* message length in BYTES */ 1967 1968 /* 1969 * In multi-byte mode, we have to keep track of where we are. 1970 * The first bytes of multi-byte chars get the full count for the 1971 * whole character. We don't do any column calculations 1972 * here, but we need the information for when we do. We could 1973 * come across cases where we are getting garbage on the 1974 * line, but we're in multi-byte mode. In that case, we may 1975 * see ASCII controls come in the middle of what should have been a 1976 * multi-byte character. Call ldterm_eucwarn...eventually, a 1977 * warning message will be printed about it. 1978 */ 1979 if (tp->t_state & TS_MEUC) { 1980 if (tp->t_eucleft) { /* if in a multi-byte char already */ 1981 --tp->t_eucleft; 1982 *tp->t_eucp++ = 0; /* is a subsequent byte */ 1983 if (c < (uchar_t)0x20) 1984 ldterm_eucwarn(tp); 1985 } else { /* is the first byte of a multi-byte, or is ASCII */ 1986 if (ISASCII(c)) { 1987 *tp->t_eucp++ = 1988 tp->t_csmethods.ldterm_dispwidth(c, 1989 (void *)tp, tp->t_modes.c_lflag & ECHOCTL); 1990 tp->t_codeset = 0; 1991 } else { 1992 *tp->t_eucp++ = 1993 tp->t_csmethods.ldterm_dispwidth(c, 1994 (void *)tp, tp->t_modes.c_lflag & ECHOCTL); 1995 tp->t_eucleft = 1996 tp->t_csmethods.ldterm_memwidth(c, 1997 (void *)tp) - 1; 1998 tp->t_codeset = ldterm_codeset( 1999 tp->t_csdata.codeset_type, c); 2000 } 2001 } 2002 } 2003 /* 2004 * AT&T is concerned about the following but we aren't since 2005 * we have already shipped code that works. 2006 * 2007 * EOL2/XCASE should be conditioned with IEXTEN to be truly 2008 * POSIX conformant. This is going to cause problems for 2009 * pre-SVR4.0 programs that don't know about IEXTEN. Hence 2010 * EOL2/IEXTEN is not conditioned with IEXTEN. 2011 */ 2012 if (!(tp->t_state & TS_SLNCH) && 2013 (c == '\n' || (c != '\0' && (c == tp->t_modes.c_cc[VEOL] || 2014 (c == tp->t_modes.c_cc[VEOL2]))))) { 2015 /* 2016 * || ((tp->t_modes.c_lflag & IEXTEN) && c == 2017 * tp->t_modes.c_cc[VEOL2]))))) { 2018 */ 2019 /* 2020 * It's a line-termination character; send the line 2021 * up the stream. 2022 */ 2023 bpt->b_datap->db_type = M_DATA; 2024 ldterm_msg_upstream(q, tp); 2025 if (tp->t_state & TS_MEUC) { 2026 ASSERT(tp->t_eucp_mp); 2027 tp->t_eucp = tp->t_eucp_mp->b_rptr; 2028 } 2029 if ((bpt = newmsg(tp)) == NULL) 2030 goto out; 2031 } else { 2032 /* 2033 * Character was escaped with LNEXT. 2034 */ 2035 if (tp->t_rocount++ == 0) 2036 tp->t_rocol = tp->t_col; 2037 tp->t_state &= ~(TS_SLNCH|TS_QUOT); 2038 /* 2039 * If the current character is a single byte and single 2040 * column character and it is the backslash character and 2041 * IEXTEN, then the state will have TS_QUOT. 2042 */ 2043 if ((c == '\\') && (tp->t_modes.c_lflag & IEXTEN) && 2044 (!(tp->t_state & TS_MEUC) || 2045 ((tp->t_state & TS_MEUC) && (!tp->t_eucleft)))) 2046 tp->t_state |= TS_QUOT; 2047 } 2048 2049 /* 2050 * Echo it. 2051 */ 2052 if (tp->t_state & TS_ERASE) { 2053 tp->t_state &= ~TS_ERASE; 2054 if (tp->t_modes.c_lflag & ECHO) 2055 ldterm_outchar('/', wrq, ebsize, tp); 2056 } 2057 if (tp->t_modes.c_lflag & ECHO) 2058 (void) ldterm_echo(c, wrq, ebsize, tp); 2059 else { 2060 /* 2061 * Echo NL when ECHO turned off, if ECHONL flag is 2062 * set. 2063 */ 2064 if (c == '\n' && (tp->t_modes.c_lflag & ECHONL)) 2065 ldterm_outchar(c, wrq, ebsize, tp); 2066 } 2067 2068 out: 2069 2070 return (bpt); 2071 } 2072 2073 2074 static int 2075 ldterm_unget(ldtermstd_state_t *tp) 2076 { 2077 mblk_t *bpt; 2078 2079 if ((bpt = tp->t_endmsg) == NULL) 2080 return (-1); /* no buffers */ 2081 if (bpt->b_rptr == bpt->b_wptr) 2082 return (-1); /* zero-length record */ 2083 tp->t_msglen--; /* one fewer character */ 2084 return (*--bpt->b_wptr); 2085 } 2086 2087 2088 static void 2089 ldterm_trim(ldtermstd_state_t *tp) 2090 { 2091 mblk_t *bpt; 2092 mblk_t *bp; 2093 2094 ASSERT(tp->t_endmsg); 2095 bpt = tp->t_endmsg; 2096 2097 if (bpt->b_rptr == bpt->b_wptr) { 2098 /* 2099 * This mblk is now empty. Find the previous mblk; 2100 * throw this one away, unless it's the first one. 2101 */ 2102 bp = tp->t_message; 2103 if (bp != bpt) { 2104 while (bp->b_cont != bpt) { 2105 ASSERT(bp->b_cont); 2106 bp = bp->b_cont; 2107 } 2108 bp->b_cont = NULL; 2109 freeb(bpt); 2110 tp->t_endmsg = bp; /* point to that mblk */ 2111 } 2112 } 2113 } 2114 2115 2116 /* 2117 * Rubout one character from the current line being built for tp as 2118 * cleanly as possible. q is the write queue for tp. Most of this 2119 * can't be applied to multi-byte processing. We do our own thing 2120 * for that... See the "ldterm_eucerase" routine. We never call 2121 * ldterm_rubout on a multi-byte or multi-column character. 2122 */ 2123 static void 2124 ldterm_rubout(uchar_t c, queue_t *q, size_t ebsize, ldtermstd_state_t *tp) 2125 { 2126 int tabcols; 2127 static unsigned char crtrubout[] = "\b \b\b \b"; 2128 #define RUBOUT1 &crtrubout[3] /* rub out one position */ 2129 #define RUBOUT2 &crtrubout[0] /* rub out two positions */ 2130 2131 if (!(tp->t_modes.c_lflag & ECHO)) 2132 return; 2133 if (tp->t_modes.c_lflag & ECHOE) { 2134 /* 2135 * "CRT rubout"; try erasing it from the screen. 2136 */ 2137 if (tp->t_rocount == 0) { 2138 /* 2139 * After the character being erased was 2140 * echoed, some data was written to the 2141 * terminal; we can't erase it cleanly, so we 2142 * just reprint the whole line as if the user 2143 * had typed the reprint character. 2144 */ 2145 ldterm_reprint(q, ebsize, tp); 2146 return; 2147 } else { 2148 /* 2149 * XXX what about escaped characters? 2150 */ 2151 switch (typetab[c]) { 2152 2153 case ORDINARY: 2154 if ((tp->t_modes.c_lflag & XCASE) && 2155 omaptab[c]) 2156 ldterm_outstring(RUBOUT1, 3, q, ebsize, 2157 tp); 2158 ldterm_outstring(RUBOUT1, 3, q, ebsize, tp); 2159 break; 2160 2161 case VTAB: 2162 case BACKSPACE: 2163 case CONTROL: 2164 case RETURN: 2165 case NEWLINE: 2166 if ((tp->t_modes.c_lflag & ECHOCTL) && 2167 (tp->t_modes.c_lflag & IEXTEN)) 2168 ldterm_outstring(RUBOUT2, 6, q, ebsize, 2169 tp); 2170 break; 2171 2172 case TAB: 2173 if (tp->t_rocount < tp->t_msglen) { 2174 /* 2175 * While the tab being erased was 2176 * expanded, some data was written 2177 * to the terminal; we can't erase 2178 * it cleanly, so we just reprint 2179 * the whole line as if the user 2180 * had typed the reprint character. 2181 */ 2182 ldterm_reprint(q, ebsize, tp); 2183 return; 2184 } 2185 tabcols = ldterm_tabcols(tp); 2186 while (--tabcols >= 0) 2187 ldterm_outchar('\b', q, ebsize, tp); 2188 break; 2189 } 2190 } 2191 } else if ((tp->t_modes.c_lflag & ECHOPRT) && 2192 (tp->t_modes.c_lflag & IEXTEN)) { 2193 /* 2194 * "Printing rubout"; echo it between \ and /. 2195 */ 2196 if (!(tp->t_state & TS_ERASE)) { 2197 ldterm_outchar('\\', q, ebsize, tp); 2198 tp->t_state |= TS_ERASE; 2199 } 2200 (void) ldterm_echo(c, q, ebsize, tp); 2201 } else 2202 (void) ldterm_echo(tp->t_modes.c_cc[VERASE], q, ebsize, tp); 2203 tp->t_rocount--; /* we "unechoed" this character */ 2204 } 2205 2206 2207 /* 2208 * Find the number of characters the tab we just deleted took up by 2209 * zipping through the current line and recomputing the column 2210 * number. 2211 */ 2212 static int 2213 ldterm_tabcols(ldtermstd_state_t *tp) 2214 { 2215 int col; 2216 int i; 2217 mblk_t *bp; 2218 unsigned char *readp, *endp; 2219 unsigned char c; 2220 uchar_t *startp; 2221 char errflg; 2222 uchar_t u8[LDTERM_CS_MAX_BYTE_LENGTH]; 2223 2224 col = tp->t_rocol; 2225 /* 2226 * If we're doing multi-byte stuff, zip through the list of 2227 * widths to figure out where we are (we've kept track in most 2228 * cases). 2229 */ 2230 if (tp->t_state & TS_MEUC) { 2231 ASSERT(tp->t_eucp_mp); 2232 bp = tp->t_message; 2233 startp = bp->b_datap->db_base; 2234 readp = tp->t_eucp_mp->b_rptr; 2235 endp = tp->t_eucp; 2236 errflg = (char)0; 2237 while (readp < endp) { 2238 switch (*readp) { 2239 case EUC_TWIDTH: /* it's a tab */ 2240 col |= 07; /* bump up */ 2241 col++; 2242 break; 2243 case EUC_BSWIDTH: /* backspace */ 2244 if (col) 2245 col--; 2246 break; 2247 case EUC_NLWIDTH: /* newline */ 2248 if (tp->t_modes.c_oflag & ONLRET) 2249 col = 0; 2250 break; 2251 case EUC_CRWIDTH: /* return */ 2252 col = 0; 2253 break; 2254 case UNKNOWN_WIDTH: /* UTF-8 unknown width */ 2255 if (tp->t_csdata.codeset_type != 2256 LDTERM_CS_TYPE_UTF8 || errflg) { 2257 *readp = 1; 2258 col++; 2259 break; 2260 } 2261 /* 2262 * Collect the current UTF-8 character bytes 2263 * from (possibly multiple) data buffers so 2264 * that we can figure out the display width. 2265 */ 2266 u8[0] = *startp; 2267 for (i = 1; (i < LDTERM_CS_MAX_BYTE_LENGTH) && 2268 (*(readp + i) == 0); i++) { 2269 startp++; 2270 if (startp >= bp->b_datap->db_lim) { 2271 if (bp->b_cont) { 2272 bp = bp->b_cont; 2273 startp = 2274 bp->b_datap-> 2275 db_base; 2276 } else { 2277 *readp = 1; 2278 col++; 2279 break; 2280 } 2281 } 2282 u8[i] = *startp; 2283 } 2284 2285 /* tp->t_eucp_mp contains wrong info?? */ 2286 if (*readp == 1) 2287 break; 2288 2289 *readp = ldterm_utf8_width(u8, i); 2290 2291 col += *readp; 2292 readp += (i - 1); 2293 break; 2294 default: 2295 col += *readp; 2296 break; 2297 } 2298 ++readp; 2299 ++startp; 2300 if (startp >= bp->b_datap->db_lim) { 2301 if (bp->b_cont) { 2302 bp = bp->b_cont; 2303 startp = bp->b_datap->db_base; 2304 } else { 2305 /* 2306 * This will happen only if 2307 * tp->t_eucp_mp contains wrong 2308 * display width info. 2309 */ 2310 errflg = (char)1; 2311 startp--; 2312 } 2313 } 2314 } 2315 goto eucout; /* finished! */ 2316 } 2317 bp = tp->t_message; 2318 do { 2319 readp = bp->b_rptr; 2320 while (readp < bp->b_wptr) { 2321 c = *readp++; 2322 if ((tp->t_modes.c_lflag & ECHOCTL) && 2323 (tp->t_modes.c_lflag & IEXTEN)) { 2324 if (c <= 037 && c != '\t' && c != '\n' || 2325 c == 0177) { 2326 col += 2; 2327 continue; 2328 } 2329 } 2330 /* 2331 * Column position calculated here. 2332 */ 2333 switch (typetab[c]) { 2334 2335 /* 2336 * Ordinary characters; advance by 2337 * one. 2338 */ 2339 case ORDINARY: 2340 col++; 2341 break; 2342 2343 /* 2344 * Non-printing characters; nothing 2345 * happens. 2346 */ 2347 case CONTROL: 2348 break; 2349 2350 /* Backspace */ 2351 case BACKSPACE: 2352 if (col != 0) 2353 col--; 2354 break; 2355 2356 /* Newline; column depends on flags. */ 2357 case NEWLINE: 2358 if (tp->t_modes.c_oflag & ONLRET) 2359 col = 0; 2360 break; 2361 2362 /* tab */ 2363 case TAB: 2364 col |= 07; 2365 col++; 2366 break; 2367 2368 /* vertical motion */ 2369 case VTAB: 2370 break; 2371 2372 /* carriage return */ 2373 case RETURN: 2374 col = 0; 2375 break; 2376 } 2377 } 2378 } while ((bp = bp->b_cont) != NULL); /* next block, if any */ 2379 2380 /* 2381 * "col" is now the column number before the tab. "tp->t_col" 2382 * is still the column number after the tab, since we haven't 2383 * erased the tab yet. Thus "tp->t_col - col" is the number 2384 * of positions the tab moved. 2385 */ 2386 eucout: 2387 col = tp->t_col - col; 2388 if (col > 8) 2389 col = 8; /* overflow screw */ 2390 return (col); 2391 } 2392 2393 2394 /* 2395 * Erase a single character; We ONLY ONLY deal with ASCII or 2396 * single-column single-byte codeset character. For multi-byte characters, 2397 * see "ldterm_csi_erase". 2398 */ 2399 static void 2400 ldterm_erase(queue_t *q, size_t ebsize, ldtermstd_state_t *tp) 2401 { 2402 int c; 2403 2404 if ((c = ldterm_unget(tp)) != -1) { 2405 ldterm_rubout((unsigned char) c, q, ebsize, tp); 2406 ldterm_trim(tp); 2407 if (tp->t_state & TS_MEUC) 2408 --tp->t_eucp; 2409 } 2410 } 2411 2412 2413 /* 2414 * Erase an entire word, single-byte EUC only please. 2415 */ 2416 static void 2417 ldterm_werase(queue_t *q, size_t ebsize, ldtermstd_state_t *tp) 2418 { 2419 int c; 2420 2421 /* 2422 * Erase trailing white space, if any. 2423 */ 2424 while ((c = ldterm_unget(tp)) == ' ' || c == '\t') { 2425 ldterm_rubout((unsigned char) c, q, ebsize, tp); 2426 ldterm_trim(tp); 2427 } 2428 2429 /* 2430 * Erase non-white-space characters, if any. 2431 */ 2432 while (c != -1 && c != ' ' && c != '\t') { 2433 ldterm_rubout((unsigned char) c, q, ebsize, tp); 2434 ldterm_trim(tp); 2435 c = ldterm_unget(tp); 2436 } 2437 if (c != -1) { 2438 /* 2439 * We removed one too many characters; put the last 2440 * one back. 2441 */ 2442 tp->t_endmsg->b_wptr++; /* put 'c' back */ 2443 tp->t_msglen++; 2444 } 2445 } 2446 2447 2448 /* 2449 * ldterm_csi_werase - This is multi-byte equivalent of "word erase". 2450 * "Word erase" only makes sense in languages which space between words, 2451 * and it's presumptuous for us to attempt "word erase" when we don't 2452 * know anything about what's really going on. It makes no sense for 2453 * many languages, as the criteria for defining words and tokens may 2454 * be completely different. 2455 * 2456 * In the TS_MEUC case (which is how we got here), we define a token to 2457 * be space- or tab-delimited, and erase one of them. It helps to 2458 * have this for command lines, but it's otherwise useless for text 2459 * editing applications; you need more sophistication than we can 2460 * provide here. 2461 */ 2462 static void 2463 ldterm_csi_werase(queue_t *q, size_t ebsize, ldtermstd_state_t *tp) 2464 { 2465 int c, i; 2466 int len; 2467 uchar_t *ip; 2468 uchar_t u8[LDTERM_CS_MAX_BYTE_LENGTH]; 2469 uchar_t u8_2[LDTERM_CS_MAX_BYTE_LENGTH]; 2470 2471 /* 2472 * ip points to the width of the actual bytes. t_eucp points 2473 * one byte beyond, where the next thing will be inserted. 2474 */ 2475 ip = tp->t_eucp - 1; 2476 /* 2477 * Erase trailing white space, if any. 2478 */ 2479 while ((c = ldterm_unget(tp)) == ' ' || c == '\t') { 2480 tp->t_eucp--; 2481 ldterm_rubout((unsigned char) c, q, ebsize, tp); 2482 ldterm_trim(tp); 2483 --ip; 2484 } 2485 2486 /* 2487 * Erase non-white-space characters, if any. The outer loop 2488 * bops through each byte in the buffer. Multi-byte is removed, as 2489 * is ASCII, one byte at a time. The inner loop (for) is only 2490 * executed for first bytes of multi-byte. The inner loop erases 2491 * the number of columns required for the multi-byte char. We check 2492 * for ASCII first, and ldterm_rubout knows about ASCII. 2493 */ 2494 len = 0; 2495 while (c != -1 && c != ' ' && c != '\t') { 2496 tp->t_eucp--; 2497 if (len < LDTERM_CS_MAX_BYTE_LENGTH) { 2498 u8[len++] = (uchar_t)c; 2499 } 2500 /* 2501 * Unlike EUC, except the leading byte, some bytes of 2502 * a non-EUC multi-byte characters are in the ASCII code 2503 * range, esp., 0x41 ~ 0x7a. Thus, we cannot simply check 2504 * ISASCII(). 2505 * Checking the (*ip == 1 || *ip == 2 || *ip > UNKNOWN_WIDTH) 2506 * will ensure that it is a single byte character (even though 2507 * it is on display width not byte length) and can be further 2508 * checked whether it is an ASCII character or not. 2509 * 2510 * When ECHOCTL is on and 'c' is an ASCII control character, 2511 * *ip == 2 happens. 2512 */ 2513 if ((*ip == 1 || *ip == 2 || *ip > UNKNOWN_WIDTH) && 2514 ISASCII(c)) { 2515 ldterm_rubout((unsigned char) c, q, ebsize, tp); 2516 len = 0; 2517 } else if (*ip) { 2518 if (*ip == UNKNOWN_WIDTH) { 2519 if (tp->t_csdata.codeset_type == 2520 LDTERM_CS_TYPE_UTF8) { 2521 for (i = 0; i < len; i++) 2522 u8_2[i] = u8[len - i - 1]; 2523 *ip = ldterm_utf8_width(u8_2, len); 2524 } else { 2525 *ip = 1; 2526 } 2527 } 2528 /* 2529 * erase for number of columns required for 2530 * this multi-byte character. Hopefully, matches 2531 * ldterm_dispwidth! 2532 */ 2533 for (i = 0; i < (int)*ip; i++) 2534 ldterm_rubout(' ', q, ebsize, tp); 2535 len = 0; 2536 } 2537 ldterm_trim(tp); 2538 --ip; 2539 c = ldterm_unget(tp); 2540 } 2541 if (c != -1) { 2542 /* 2543 * We removed one too many characters; put the last 2544 * one back. 2545 */ 2546 tp->t_endmsg->b_wptr++; /* put 'c' back */ 2547 tp->t_msglen++; 2548 } 2549 } 2550 2551 2552 /* 2553 * Kill an entire line, erasing each character one-by-one (if ECHOKE 2554 * is set) or just echoing the kill character, followed by a newline 2555 * (if ECHOK is set). Multi-byte processing is included here. 2556 */ 2557 2558 static void 2559 ldterm_kill(queue_t *q, size_t ebsize, ldtermstd_state_t *tp) 2560 { 2561 int c, i; 2562 int len; 2563 uchar_t *ip; 2564 uchar_t u8[LDTERM_CS_MAX_BYTE_LENGTH]; 2565 uchar_t u8_2[LDTERM_CS_MAX_BYTE_LENGTH]; 2566 2567 if ((tp->t_modes.c_lflag & ECHOKE) && 2568 (tp->t_modes.c_lflag & IEXTEN) && 2569 (tp->t_msglen == tp->t_rocount)) { 2570 if (tp->t_state & TS_MEUC) { 2571 ip = tp->t_eucp - 1; 2572 /* 2573 * This loop similar to "ldterm_csi_werase" above. 2574 */ 2575 len = 0; 2576 while ((c = ldterm_unget(tp)) != (-1)) { 2577 tp->t_eucp--; 2578 if (len < LDTERM_CS_MAX_BYTE_LENGTH) { 2579 u8[len++] = (uchar_t)c; 2580 } 2581 if ((*ip == 1 || *ip == 2 || 2582 *ip > UNKNOWN_WIDTH) && ISASCII(c)) { 2583 ldterm_rubout((unsigned char) c, q, 2584 ebsize, tp); 2585 len = 0; 2586 } else if (*ip) { 2587 if (*ip == UNKNOWN_WIDTH) { 2588 if (tp->t_csdata.codeset_type 2589 == LDTERM_CS_TYPE_UTF8) { 2590 for (i = 0; i < len; 2591 i++) 2592 u8_2[i] = 2593 u8[len-i-1]; 2594 *ip = ldterm_utf8_width( 2595 u8_2, len); 2596 } else { 2597 *ip = 1; 2598 } 2599 } 2600 for (i = 0; i < (int)*ip; i++) 2601 ldterm_rubout(' ', q, ebsize, 2602 tp); 2603 len = 0; 2604 } 2605 ldterm_trim(tp); 2606 --ip; 2607 } 2608 } else { 2609 while ((c = ldterm_unget(tp)) != -1) { 2610 ldterm_rubout((unsigned char) c, q, ebsize, tp); 2611 ldterm_trim(tp); 2612 } 2613 } 2614 } else { 2615 (void) ldterm_echo(tp->t_modes.c_cc[VKILL], q, ebsize, tp); 2616 if (tp->t_modes.c_lflag & ECHOK) 2617 (void) ldterm_echo('\n', q, ebsize, tp); 2618 while (ldterm_unget(tp) != -1) { 2619 if (tp->t_state & TS_MEUC) 2620 --tp->t_eucp; 2621 ldterm_trim(tp); 2622 } 2623 tp->t_rocount = 0; 2624 if (tp->t_state & TS_MEUC) 2625 tp->t_eucp = tp->t_eucp_mp->b_rptr; 2626 } 2627 tp->t_state &= ~(TS_QUOT|TS_ERASE|TS_SLNCH); 2628 } 2629 2630 2631 /* 2632 * Reprint the current input line. We assume c_cc has already been 2633 * checked. XXX just the current line, not the whole queue? What 2634 * about DEFECHO mode? 2635 */ 2636 static void 2637 ldterm_reprint(queue_t *q, size_t ebsize, ldtermstd_state_t *tp) 2638 { 2639 mblk_t *bp; 2640 unsigned char *readp; 2641 2642 if (tp->t_modes.c_cc[VREPRINT] != (unsigned char) 0) 2643 (void) ldterm_echo(tp->t_modes.c_cc[VREPRINT], q, ebsize, tp); 2644 ldterm_outchar('\n', q, ebsize, tp); 2645 2646 bp = tp->t_message; 2647 do { 2648 readp = bp->b_rptr; 2649 while (readp < bp->b_wptr) 2650 (void) ldterm_echo(*readp++, q, ebsize, tp); 2651 } while ((bp = bp->b_cont) != NULL); /* next block, if any */ 2652 2653 tp->t_state &= ~TS_ERASE; 2654 tp->t_rocount = tp->t_msglen; /* we reechoed the entire line */ 2655 tp->t_rocol = 0; 2656 } 2657 2658 2659 /* 2660 * Non canonical processing. Called with q locked from ldtermrsrv. 2661 * 2662 */ 2663 static mblk_t * 2664 ldterm_dononcanon(mblk_t *bp, mblk_t *bpt, size_t ebsize, queue_t *q, 2665 ldtermstd_state_t *tp) 2666 { 2667 queue_t *wrq = WR(q); 2668 unsigned char *rptr; 2669 size_t bytes_in_bp; 2670 size_t roomleft; 2671 size_t bytes_to_move; 2672 int free_flag = 0; 2673 2674 if (tp->t_modes.c_lflag & (ECHO|ECHONL|IEXTEN)) { 2675 unsigned char *wptr; 2676 unsigned char c; 2677 2678 /* 2679 * Either we must echo the characters, or we must 2680 * echo NL, or we must check for VLNEXT. Process 2681 * characters one at a time. 2682 */ 2683 rptr = bp->b_rptr; 2684 wptr = bp->b_rptr; 2685 while (rptr < bp->b_wptr) { 2686 c = *rptr++; 2687 /* 2688 * If this character is the literal next 2689 * character, echo it as '^' and backspace 2690 * over it if echoing is enabled, indicate 2691 * that the next character is to be treated 2692 * literally, and remove the LNEXT from the 2693 * input stream. 2694 * 2695 * If the *previous* character was the literal 2696 * next character, don't check whether this 2697 * is a literal next or not. 2698 */ 2699 if ((tp->t_modes.c_lflag & IEXTEN) && 2700 !(tp->t_state & TS_SLNCH) && 2701 c != _POSIX_VDISABLE && 2702 c == tp->t_modes.c_cc[VLNEXT]) { 2703 if (tp->t_modes.c_lflag & ECHO) 2704 ldterm_outstring( 2705 (unsigned char *)"^\b", 2706 2, wrq, ebsize, tp); 2707 tp->t_state |= TS_SLNCH; 2708 continue; /* and ignore it */ 2709 } 2710 /* 2711 * Not a "literal next" character, so it 2712 * should show up as input. If it was 2713 * literal-nexted, turn off the literal-next 2714 * flag. 2715 */ 2716 tp->t_state &= ~TS_SLNCH; 2717 *wptr++ = c; 2718 if (tp->t_modes.c_lflag & ECHO) { 2719 /* 2720 * Echo the character. 2721 */ 2722 (void) ldterm_echo(c, wrq, ebsize, tp); 2723 } else if (tp->t_modes.c_lflag & ECHONL) { 2724 /* 2725 * Echo NL, even though ECHO is not 2726 * set. 2727 */ 2728 if (c == '\n') 2729 ldterm_outchar('\n', wrq, 1, tp); 2730 } 2731 } 2732 bp->b_wptr = wptr; 2733 } else { 2734 /* 2735 * If there are any characters in this buffer, and 2736 * the first of them was literal-nexted, turn off the 2737 * literal-next flag. 2738 */ 2739 if (bp->b_rptr != bp->b_wptr) 2740 tp->t_state &= ~TS_SLNCH; 2741 } 2742 2743 ASSERT(bp->b_wptr >= bp->b_rptr); 2744 bytes_in_bp = bp->b_wptr - bp->b_rptr; 2745 rptr = bp->b_rptr; 2746 while (bytes_in_bp != 0) { 2747 roomleft = bpt->b_datap->db_lim - bpt->b_wptr; 2748 if (roomleft == 0) { 2749 /* 2750 * No more room in this mblk; save this one 2751 * away, and allocate a new one. 2752 */ 2753 if ((bpt = allocb(IBSIZE, BPRI_MED)) == NULL) { 2754 freeb(bp); 2755 DEBUG4(("ldterm_do_noncanon: allcob failed\n")); 2756 return (bpt); 2757 } 2758 /* 2759 * Chain the new one to the end of the old 2760 * one, and mark it as the last block in the 2761 * current lump. 2762 */ 2763 tp->t_endmsg->b_cont = bpt; 2764 tp->t_endmsg = bpt; 2765 roomleft = IBSIZE; 2766 } 2767 DEBUG5(("roomleft=%d, bytes_in_bp=%d, tp->t_rd_request=%d\n", 2768 roomleft, bytes_in_bp, tp->t_rd_request)); 2769 /* 2770 * if there is a read pending before this data got 2771 * here move bytes according to the minimum of room 2772 * left in this buffer, bytes in the message and byte 2773 * count requested in the read. If there is no read 2774 * pending, move the minimum of the first two 2775 */ 2776 if (tp->t_rd_request == 0) 2777 bytes_to_move = MIN(roomleft, bytes_in_bp); 2778 else 2779 bytes_to_move = 2780 MIN(MIN(roomleft, bytes_in_bp), tp->t_rd_request); 2781 DEBUG5(("Bytes to move = %lu\n", bytes_to_move)); 2782 if (bytes_to_move == 0) 2783 break; 2784 bcopy(rptr, bpt->b_wptr, bytes_to_move); 2785 bpt->b_wptr += bytes_to_move; 2786 rptr += bytes_to_move; 2787 tp->t_msglen += bytes_to_move; 2788 bytes_in_bp -= bytes_to_move; 2789 } 2790 if (bytes_in_bp == 0) { 2791 DEBUG4(("bytes_in_bp is zero\n")); 2792 freeb(bp); 2793 } else 2794 free_flag = 1; /* for debugging olny */ 2795 2796 DEBUG4(("ldterm_do_noncanon: VMIN = %d, VTIME = %d, msglen = %d, \ 2797 tid = %d\n", V_MIN, V_TIME, tp->t_msglen, tp->t_vtid)); 2798 /* 2799 * If there is a pending read request at the stream head we 2800 * need to do VMIN/VTIME processing. The four possible cases 2801 * are: 2802 * MIN = 0, TIME > 0 2803 * MIN = >, TIME = 0 2804 * MIN > 0, TIME > 0 2805 * MIN = 0, TIME = 0 2806 * If we can satisfy VMIN, send it up, and start a new 2807 * timer if necessary. These four cases of VMIN/VTIME 2808 * are also dealt with in the write side put routine 2809 * when the M_READ is first seen. 2810 */ 2811 2812 DEBUG4(("Incoming data while M_READ'ing\n")); 2813 /* 2814 * Case 1: Any data will satisfy the read, so send 2815 * it upstream. 2816 */ 2817 if (V_MIN == 0 && V_TIME > 0) { 2818 if (tp->t_msglen) 2819 vmin_satisfied(q, tp, 1); 2820 else { 2821 /* EMPTY */ 2822 DEBUG4(("ldterm_do_noncanon called, but no data!\n")); 2823 } 2824 /* 2825 * Case 2: This should never time out, so 2826 * until there's enough data, do nothing. 2827 */ 2828 } else if (V_MIN > 0 && V_TIME == 0) { 2829 if (tp->t_msglen >= (int)V_MIN) 2830 vmin_satisfied(q, tp, 1); 2831 2832 /* 2833 * Case 3: If MIN is satisfied, send it up. 2834 * Also, remember to start a new timer *every* 2835 * time we see something if MIN isn't 2836 * safisfied 2837 */ 2838 } else if (V_MIN > 0 && V_TIME > 0) { 2839 if (tp->t_msglen >= (int)V_MIN) 2840 vmin_satisfied(q, tp, 1); 2841 else 2842 vmin_settimer(q); 2843 /* 2844 * Case 4: Not possible. This request 2845 * should always be satisfied from the write 2846 * side, left here for debugging. 2847 */ 2848 } else { /* V_MIN == 0 && V_TIME == 0 */ 2849 vmin_satisfied(q, tp, 1); 2850 } 2851 2852 if (free_flag) { 2853 /* EMPTY */ 2854 DEBUG4(("CAUTION message block not freed\n")); 2855 } 2856 return (newmsg(tp)); 2857 } 2858 2859 2860 /* 2861 * Echo a typed byte to the terminal. Returns the number of bytes 2862 * printed. Bytes of EUC characters drop through the ECHOCTL stuff 2863 * and are just output as themselves. 2864 */ 2865 static int 2866 ldterm_echo(uchar_t c, queue_t *q, size_t ebsize, ldtermstd_state_t *tp) 2867 { 2868 int i; 2869 2870 if (!(tp->t_modes.c_lflag & ECHO)) 2871 return (0); 2872 i = 0; 2873 2874 /* 2875 * Echo control characters (c <= 37) only if the ECHOCTRL 2876 * flag is set as ^X. 2877 */ 2878 2879 if ((tp->t_modes.c_lflag & ECHOCTL) && 2880 (tp->t_modes.c_lflag & IEXTEN)) { 2881 if (c <= 037 && c != '\t' && c != '\n') { 2882 ldterm_outchar('^', q, ebsize, tp); 2883 i++; 2884 if (tp->t_modes.c_oflag & OLCUC) 2885 c += 'a' - 1; 2886 else 2887 c += 'A' - 1; 2888 } else if (c == 0177) { 2889 ldterm_outchar('^', q, ebsize, tp); 2890 i++; 2891 c = '?'; 2892 } 2893 ldterm_outchar(c, q, ebsize, tp); 2894 return (i + 1); 2895 /* echo only special control character and the Bell */ 2896 } else if ((c > 037 && c != 0177) || c == '\t' || c == '\n' || 2897 c == '\r' || c == '\b' || c == 007 || 2898 c == tp->t_modes.c_cc[VKILL]) { 2899 ldterm_outchar(c, q, ebsize, tp); 2900 return (i + 1); 2901 } 2902 return (i); 2903 } 2904 2905 2906 /* 2907 * Put a character on the output queue. 2908 */ 2909 static void 2910 ldterm_outchar(uchar_t c, queue_t *q, size_t bsize, ldtermstd_state_t *tp) 2911 { 2912 mblk_t *curbp; 2913 2914 /* 2915 * Don't even look at the characters unless we have something 2916 * useful to do with them. 2917 */ 2918 if ((tp->t_modes.c_oflag & OPOST) || 2919 ((tp->t_modes.c_lflag & XCASE) && 2920 (tp->t_modes.c_lflag & ICANON))) { 2921 mblk_t *mp; 2922 2923 if ((mp = allocb(4, BPRI_HI)) == NULL) { 2924 cmn_err(CE_WARN, 2925 "ldterm: (ldterm_outchar) out of blocks"); 2926 return; 2927 } 2928 *mp->b_wptr++ = c; 2929 mp = ldterm_output_msg(q, mp, &tp->t_echomp, tp, bsize, 1); 2930 if (mp != NULL) 2931 freemsg(mp); 2932 2933 } else { 2934 if ((curbp = tp->t_echomp) != NULL) { 2935 while (curbp->b_cont != NULL) 2936 curbp = curbp->b_cont; 2937 if (curbp->b_datap->db_lim == curbp->b_wptr) { 2938 mblk_t *newbp; 2939 2940 if ((newbp = allocb(bsize, BPRI_HI)) == NULL) { 2941 cmn_err(CE_WARN, 2942 "ldterm_outchar: out of blocks"); 2943 return; 2944 } 2945 curbp->b_cont = newbp; 2946 curbp = newbp; 2947 } 2948 } else { 2949 if ((curbp = allocb(bsize, BPRI_HI)) == NULL) { 2950 cmn_err(CE_WARN, 2951 "ldterm_outchar: out of blocks"); 2952 return; 2953 } 2954 tp->t_echomp = curbp; 2955 } 2956 *curbp->b_wptr++ = c; 2957 } 2958 } 2959 2960 2961 /* 2962 * Copy a string, of length len, to the output queue. 2963 */ 2964 static void 2965 ldterm_outstring(uchar_t *cp, int len, queue_t *q, size_t bsize, 2966 ldtermstd_state_t *tp) 2967 { 2968 while (len > 0) { 2969 ldterm_outchar(*cp++, q, bsize, tp); 2970 len--; 2971 } 2972 } 2973 2974 2975 static mblk_t * 2976 newmsg(ldtermstd_state_t *tp) 2977 { 2978 mblk_t *bp; 2979 2980 /* 2981 * If no current message, allocate a block for it. 2982 */ 2983 if ((bp = tp->t_endmsg) == NULL) { 2984 if ((bp = allocb(IBSIZE, BPRI_MED)) == NULL) { 2985 cmn_err(CE_WARN, 2986 "ldterm: (ldtermrsrv/newmsg) out of blocks"); 2987 return (bp); 2988 } 2989 tp->t_message = bp; 2990 tp->t_endmsg = bp; 2991 } 2992 return (bp); 2993 } 2994 2995 2996 static void 2997 ldterm_msg_upstream(queue_t *q, ldtermstd_state_t *tp) 2998 { 2999 ssize_t s; 3000 mblk_t *bp; 3001 3002 bp = tp->t_message; 3003 s = msgdsize(bp); 3004 if (bp) 3005 putnext(q, tp->t_message); 3006 3007 /* 3008 * update sysinfo canch character. 3009 */ 3010 if (CANON_MODE) 3011 (void) drv_setparm(SYSCANC, s); 3012 tp->t_message = NULL; 3013 tp->t_endmsg = NULL; 3014 tp->t_msglen = 0; 3015 tp->t_rocount = 0; 3016 tp->t_rd_request = 0; 3017 if (tp->t_state & TS_MEUC) { 3018 ASSERT(tp->t_eucp_mp); 3019 tp->t_eucp = tp->t_eucp_mp->b_rptr; 3020 /* can't reset everything, as we may have other input */ 3021 } 3022 } 3023 3024 3025 /* 3026 * Re-enable the write-side service procedure. When an allocation 3027 * failure causes write-side processing to stall, we disable the 3028 * write side and arrange to call this function when allocation once 3029 * again becomes possible. 3030 */ 3031 static void 3032 ldterm_wenable(void *addr) 3033 { 3034 queue_t *q = addr; 3035 ldtermstd_state_t *tp; 3036 3037 tp = (ldtermstd_state_t *)q->q_ptr; 3038 /* 3039 * The bufcall is no longer pending. 3040 */ 3041 tp->t_wbufcid = 0; 3042 enableok(q); 3043 qenable(q); 3044 } 3045 3046 3047 /* 3048 * Line discipline output queue put procedure. Attempts to process 3049 * the message directly and send it on downstream, queueing it only 3050 * if there's already something pending or if its downstream neighbor 3051 * is clogged. 3052 */ 3053 static void 3054 ldtermwput(queue_t *q, mblk_t *mp) 3055 { 3056 ldtermstd_state_t *tp; 3057 unsigned char type = mp->b_datap->db_type; 3058 3059 tp = (ldtermstd_state_t *)q->q_ptr; 3060 3061 /* 3062 * Always process priority messages, regardless of whether or 3063 * not our queue is nonempty. 3064 */ 3065 if (type >= QPCTL) { 3066 switch (type) { 3067 3068 case M_FLUSH: 3069 /* 3070 * Get rid of it, see comment in 3071 * ldterm_dosig(). 3072 */ 3073 if ((tp->t_state & TS_FLUSHWAIT) && 3074 (*mp->b_rptr == FLUSHW)) { 3075 tp->t_state &= ~TS_FLUSHWAIT; 3076 freemsg(mp); 3077 return; 3078 } 3079 /* 3080 * This is coming from above, so we only 3081 * handle the write queue here. If FLUSHR is 3082 * set, it will get turned around at the 3083 * driver, and the read procedure will see it 3084 * eventually. 3085 */ 3086 if (*mp->b_rptr & FLUSHW) { 3087 if ((tp->t_state & TS_ISPTSTTY) && 3088 (*mp->b_rptr & FLUSHBAND)) 3089 flushband(q, *(mp->b_rptr + 1), 3090 FLUSHDATA); 3091 else 3092 flushq(q, FLUSHDATA); 3093 } 3094 3095 putnext(q, mp); 3096 /* 3097 * If a timed read is interrupted, there is 3098 * no way to cancel an existing M_READ 3099 * request. We kludge by allowing a flush to 3100 * do so. 3101 */ 3102 if (tp->t_state & TS_MREAD) 3103 vmin_satisfied(RD(q), tp, 0); 3104 break; 3105 3106 case M_READ: 3107 DEBUG1(("ldtermwmsg:M_READ RECEIVED\n")); 3108 /* 3109 * Stream head needs data to satisfy timed 3110 * read. Has meaning only if ICANON flag is 3111 * off indicating raw mode 3112 */ 3113 3114 DEBUG4(( 3115 "M_READ: RAW_MODE=%d, CNT=%d, VMIN=%d, VTIME=%d\n", 3116 RAW_MODE, *(unsigned int *)mp->b_rptr, V_MIN, 3117 V_TIME)); 3118 3119 tp->t_rd_request = *(unsigned int *)mp->b_rptr; 3120 3121 if (RAW_MODE) { 3122 if (newmsg(tp) != NULL) { 3123 /* 3124 * VMIN/VTIME processing... 3125 * The four possible cases are: 3126 * MIN = 0, TIME > 0 3127 * MIN = >, TIME = 0 3128 * MIN > 0, TIME > 0 3129 * MIN = 0, TIME = 0 3130 * These four conditions must be dealt 3131 * with on the read side as well in 3132 * ldterm_do_noncanon(). Set TS_MREAD 3133 * so that the read side will know 3134 * there is a pending read request 3135 * waiting at the stream head. If we 3136 * can satisfy MIN do it here, rather 3137 * than on the read side. If we can't, 3138 * start timers if necessary and let 3139 * the other side deal with it. 3140 * 3141 * We got another M_READ before the 3142 * pending one completed, cancel any 3143 * existing timeout. 3144 */ 3145 if (tp->t_state & TS_MREAD) { 3146 vmin_satisfied(RD(q), 3147 tp, 0); 3148 } 3149 tp->t_state |= TS_MREAD; 3150 /* 3151 * Case 1: Any data will 3152 * satisfy read, otherwise 3153 * start timer 3154 */ 3155 if (V_MIN == 0 && V_TIME > 0) { 3156 if (tp->t_msglen) 3157 vmin_satisfied(RD(q), 3158 tp, 1); 3159 else 3160 vmin_settimer(RD(q)); 3161 3162 /* 3163 * Case 2: If we have enough 3164 * data, send up now. 3165 * Otherwise, the read side 3166 * should wait forever until MIN 3167 * is satisified. 3168 */ 3169 } else if (V_MIN > 0 && V_TIME == 0) { 3170 if (tp->t_msglen >= (int)V_MIN) 3171 vmin_satisfied(RD(q), 3172 tp, 1); 3173 3174 /* 3175 * Case 3: If we can satisfy 3176 * the read, send it up. If we 3177 * don't have enough data, but 3178 * there is at least one char, 3179 * start a timer. Otherwise, 3180 * let the read side start 3181 * the timer. 3182 */ 3183 } else if (V_MIN > 0 && V_TIME > 0) { 3184 if (tp->t_msglen >= (int)V_MIN) 3185 vmin_satisfied(RD(q), 3186 tp, 1); 3187 else if (tp->t_msglen) 3188 vmin_settimer(RD(q)); 3189 /* 3190 * Case 4: Read returns 3191 * whatever data is available 3192 * or zero if none. 3193 */ 3194 } else { /* V_MIN == 0 && V_TIME == 0 */ 3195 vmin_satisfied(RD(q), tp, 1); 3196 } 3197 3198 } else /* should do bufcall, really! */ 3199 cmn_err(CE_WARN, 3200 "ldtermwmsg: out of blocks"); 3201 } 3202 /* 3203 * pass M_READ down 3204 */ 3205 putnext(q, mp); 3206 break; 3207 3208 default: 3209 /* Pass it through unmolested. */ 3210 putnext(q, mp); 3211 break; 3212 } 3213 return; 3214 } 3215 /* 3216 * If our queue is nonempty or there's a traffic jam 3217 * downstream, this message must get in line. 3218 */ 3219 if (q->q_first != NULL || !bcanputnext(q, mp->b_band)) { 3220 /* 3221 * Exception: ioctls, except for those defined to 3222 * take effect after output has drained, should be 3223 * processed immediately. 3224 */ 3225 if (type == M_IOCTL) { 3226 struct iocblk *iocp; 3227 3228 iocp = (struct iocblk *)mp->b_rptr; 3229 switch (iocp->ioc_cmd) { 3230 3231 /* 3232 * Queue these. 3233 */ 3234 case TCSETSW: 3235 case TCSETSF: 3236 case TCSETAW: 3237 case TCSETAF: 3238 case TCSBRK: 3239 break; 3240 3241 /* 3242 * Handle all others immediately. 3243 */ 3244 default: 3245 (void) ldtermwmsg(q, mp); 3246 return; 3247 } 3248 } 3249 (void) putq(q, mp); 3250 return; 3251 } 3252 /* 3253 * We can take the fast path through, by simply calling 3254 * ldtermwmsg to dispose of mp. 3255 */ 3256 (void) ldtermwmsg(q, mp); 3257 } 3258 3259 3260 /* 3261 * Line discipline output queue service procedure. 3262 */ 3263 static void 3264 ldtermwsrv(queue_t *q) 3265 { 3266 mblk_t *mp; 3267 3268 /* 3269 * We expect this loop to iterate at most once, but must be 3270 * prepared for more in case our upstream neighbor isn't 3271 * paying strict attention to what canput tells it. 3272 */ 3273 while ((mp = getq(q)) != NULL) { 3274 /* 3275 * N.B.: ldtermwput has already handled high-priority 3276 * messages, so we don't have to worry about them 3277 * here. Hence, the putbq call is safe. 3278 */ 3279 if (!bcanputnext(q, mp->b_band)) { 3280 (void) putbq(q, mp); 3281 break; 3282 } 3283 if (!ldtermwmsg(q, mp)) { 3284 /* 3285 * Couldn't handle the whole thing; give up 3286 * for now and wait to be rescheduled. 3287 */ 3288 break; 3289 } 3290 } 3291 } 3292 3293 3294 /* 3295 * Process the write-side message denoted by mp. If mp can't be 3296 * processed completely (due to allocation failures), put the 3297 * residual unprocessed part on the front of the write queue, disable 3298 * the queue, and schedule a qbufcall to arrange to complete its 3299 * processing later. 3300 * 3301 * Return 1 if the message was processed completely and 0 if not. 3302 * 3303 * This routine is called from both ldtermwput and ldtermwsrv to do the 3304 * actual work of dealing with mp. ldtermwput will have already 3305 * dealt with high priority messages. 3306 */ 3307 static int 3308 ldtermwmsg(queue_t *q, mblk_t *mp) 3309 { 3310 ldtermstd_state_t *tp; 3311 mblk_t *residmp = NULL; 3312 size_t size; 3313 3314 tp = (ldtermstd_state_t *)q->q_ptr; 3315 3316 switch (mp->b_datap->db_type) { 3317 3318 case M_IOCTL: 3319 ldterm_do_ioctl(q, mp); 3320 break; 3321 3322 case M_DATA: 3323 { 3324 mblk_t *omp = NULL; 3325 3326 if ((tp->t_modes.c_lflag & FLUSHO) && 3327 (tp->t_modes.c_lflag & IEXTEN)) { 3328 freemsg(mp); /* drop on floor */ 3329 break; 3330 } 3331 tp->t_rocount = 0; 3332 /* 3333 * Don't even look at the characters unless 3334 * we have something useful to do with them. 3335 */ 3336 if (((tp->t_modes.c_oflag & OPOST) || 3337 ((tp->t_modes.c_lflag & XCASE) && 3338 (tp->t_modes.c_lflag & ICANON))) && 3339 (msgdsize(mp) || !(tp->t_state & TS_ISPTSTTY))) { 3340 unsigned char band = mp->b_band; 3341 short flag = mp->b_flag; 3342 3343 residmp = ldterm_output_msg(q, mp, &omp, 3344 tp, OBSIZE, 0); 3345 if ((mp = omp) == NULL) 3346 break; 3347 mp->b_band |= band; 3348 mp->b_flag |= flag; 3349 } 3350 /* Update sysinfo outch */ 3351 (void) drv_setparm(SYSOUTC, msgdsize(mp)); 3352 putnext(q, mp); 3353 break; 3354 } 3355 3356 default: 3357 putnext(q, mp); /* pass it through unmolested */ 3358 break; 3359 } 3360 3361 if (residmp == NULL) 3362 return (1); 3363 3364 /* 3365 * An allocation failure occurred that prevented the message 3366 * from being completely processed. First, disable our 3367 * queue, since it's pointless to attempt further processing 3368 * until the allocation situation is resolved. (This must 3369 * precede the putbq call below, which would otherwise mark 3370 * the queue to be serviced.) 3371 */ 3372 noenable(q); 3373 /* 3374 * Stuff the remnant on our write queue so that we can 3375 * complete it later when times become less lean. Note that 3376 * this sets QFULL, so that our upstream neighbor will be 3377 * blocked by flow control. 3378 */ 3379 (void) putbq(q, residmp); 3380 /* 3381 * Schedule a qbufcall to re-enable the queue. The failure 3382 * won't have been for an allocation of more than OBSIZE 3383 * bytes, so don't ask for more than that from bufcall. 3384 */ 3385 size = msgdsize(residmp); 3386 if (size > OBSIZE) 3387 size = OBSIZE; 3388 if (tp->t_wbufcid) 3389 qunbufcall(q, tp->t_wbufcid); 3390 tp->t_wbufcid = qbufcall(q, size, BPRI_MED, ldterm_wenable, q); 3391 3392 return (0); 3393 } 3394 3395 3396 /* 3397 * Perform output processing on a message, accumulating the output 3398 * characters in a new message. 3399 */ 3400 static mblk_t * 3401 ldterm_output_msg(queue_t *q, mblk_t *imp, mblk_t **omp, 3402 ldtermstd_state_t *tp, size_t bsize, int echoing) 3403 { 3404 mblk_t *ibp; /* block we're examining from input message */ 3405 mblk_t *obp; /* block we're filling in output message */ 3406 mblk_t *cbp; /* continuation block */ 3407 mblk_t *oobp; /* old value of obp; valid if NEW_BLOCK fails */ 3408 mblk_t **contpp; /* where to stuff ptr to newly-allocated blk */ 3409 unsigned char c, n; 3410 int count, ctype; 3411 ssize_t bytes_left; 3412 3413 mblk_t *bp; /* block to stuff an M_DELAY message in */ 3414 3415 3416 /* 3417 * Allocate a new block into which to put bytes. If we can't, 3418 * we just drop the rest of the message on the floor. If x is 3419 * non-zero, just fall thru; failure requires cleanup before 3420 * going out 3421 */ 3422 3423 #define NEW_BLOCK(x) \ 3424 { \ 3425 oobp = obp; \ 3426 if ((obp = allocb(bsize, BPRI_MED)) == NULL) { \ 3427 if (x == 0) \ 3428 goto outofbufs; \ 3429 } else { \ 3430 *contpp = obp; \ 3431 contpp = &obp->b_cont; \ 3432 bytes_left = obp->b_datap->db_lim - obp->b_wptr; \ 3433 } \ 3434 } 3435 3436 ibp = imp; 3437 3438 /* 3439 * When we allocate the first block of a message, we should 3440 * stuff the pointer to it in "*omp". All subsequent blocks 3441 * should have the pointer to them stuffed into the "b_cont" 3442 * field of the previous block. "contpp" points to the place 3443 * where we should stuff the pointer. 3444 * 3445 * If we already have a message we're filling in, continue doing 3446 * so. 3447 */ 3448 if ((obp = *omp) != NULL) { 3449 while (obp->b_cont != NULL) 3450 obp = obp->b_cont; 3451 contpp = &obp->b_cont; 3452 bytes_left = obp->b_datap->db_lim - obp->b_wptr; 3453 } else { 3454 contpp = omp; 3455 bytes_left = 0; 3456 } 3457 3458 do { 3459 while (ibp->b_rptr < ibp->b_wptr) { 3460 /* 3461 * Make sure there's room for one more 3462 * character. At most, we'll need "t_maxeuc" 3463 * bytes. 3464 */ 3465 if ((bytes_left < (int)tp->t_maxeuc)) { 3466 /* LINTED */ 3467 NEW_BLOCK(0); 3468 } 3469 /* 3470 * If doing XCASE processing (not very 3471 * likely, in this day and age), look at each 3472 * character individually. 3473 */ 3474 if ((tp->t_modes.c_lflag & XCASE) && 3475 (tp->t_modes.c_lflag & ICANON)) { 3476 c = *ibp->b_rptr++; 3477 3478 /* 3479 * We need to make sure that this is not 3480 * a following byte of a multibyte character 3481 * before applying an XCASE processing. 3482 * 3483 * tp->t_eucign will be 0 if and only 3484 * if the current 'c' is an ASCII character 3485 * and also a byte. Otherwise, it will have 3486 * the byte length of a multibyte character. 3487 */ 3488 if ((tp->t_state & TS_MEUC) && 3489 tp->t_eucign == 0 && NOTASCII(c)) { 3490 tp->t_eucign = 3491 tp->t_csmethods.ldterm_memwidth( 3492 c, (void *)tp); 3493 tp->t_scratch_len = tp->t_eucign; 3494 3495 if (tp->t_csdata.codeset_type != 3496 LDTERM_CS_TYPE_UTF8) { 3497 tp->t_col += 3498 tp-> 3499 t_csmethods. 3500 ldterm_dispwidth(c, 3501 (void *)tp, 3502 tp->t_modes.c_lflag & 3503 ECHOCTL); 3504 } 3505 } 3506 3507 /* 3508 * If character is mapped on output, 3509 * put out a backslash followed by 3510 * what it is mapped to. 3511 */ 3512 if (tp->t_eucign == 0 && omaptab[c] != 0 && 3513 (!echoing || c != '\\')) { 3514 /* backslash is an ordinary character */ 3515 tp->t_col++; 3516 *obp->b_wptr++ = '\\'; 3517 bytes_left--; 3518 if (bytes_left == 0) { 3519 /* LINTED */ 3520 NEW_BLOCK(1); 3521 } 3522 /* 3523 * Allocation failed, make 3524 * state consistent before 3525 * returning 3526 */ 3527 if (obp == NULL) { 3528 ibp->b_rptr--; 3529 tp->t_col--; 3530 oobp->b_wptr--; 3531 goto outofbufs; 3532 } 3533 c = omaptab[c]; 3534 } 3535 /* 3536 * If no other output processing is 3537 * required, push the character into 3538 * the block and get another. 3539 */ 3540 if (!(tp->t_modes.c_oflag & OPOST)) { 3541 if (tp->t_eucign > 0) { 3542 --tp->t_eucign; 3543 } else { 3544 tp->t_col++; 3545 } 3546 *obp->b_wptr++ = c; 3547 bytes_left--; 3548 continue; 3549 } 3550 /* 3551 * OPOST output flag is set. Map 3552 * lower case to upper case if OLCUC 3553 * flag is set and the 'c' is a lowercase 3554 * ASCII character. 3555 */ 3556 if (tp->t_eucign == 0 && 3557 (tp->t_modes.c_oflag & OLCUC) && 3558 c >= 'a' && c <= 'z') 3559 c -= 'a' - 'A'; 3560 } else { 3561 /* 3562 * Copy all the ORDINARY characters, 3563 * possibly mapping upper case to 3564 * lower case. We use "movtuc", 3565 * STOPPING when we can't move some 3566 * character. For multi-byte or 3567 * multi-column EUC, we can't depend 3568 * on the regular tables. Rather than 3569 * just drop through to the "big 3570 * switch" for all characters, it 3571 * _might_ be faster to let "movtuc" 3572 * move a bunch of characters. 3573 * Chances are, even in multi-byte 3574 * mode we'll have lots of ASCII 3575 * going through. We check the flag 3576 * once, and call movtuc with the 3577 * appropriate table as an argument. 3578 * 3579 * "movtuc will work for all codeset 3580 * types since it stops at the beginning 3581 * byte of a multibyte character. 3582 */ 3583 size_t bytes_to_move; 3584 size_t bytes_moved; 3585 3586 ASSERT(ibp->b_wptr >= ibp->b_rptr); 3587 bytes_to_move = ibp->b_wptr - ibp->b_rptr; 3588 if (bytes_to_move > bytes_left) 3589 bytes_to_move = bytes_left; 3590 if (tp->t_state & TS_MEUC) { 3591 bytes_moved = movtuc(bytes_to_move, 3592 ibp->b_rptr, obp->b_wptr, 3593 (tp->t_modes.c_oflag & OLCUC ? 3594 elcuctab : enotrantab)); 3595 } else { 3596 bytes_moved = movtuc(bytes_to_move, 3597 ibp->b_rptr, obp->b_wptr, 3598 (tp->t_modes.c_oflag & OLCUC ? 3599 lcuctab : notrantab)); 3600 } 3601 /* 3602 * We're save to just do this column 3603 * calculation, because if TS_MEUC is 3604 * set, we used the proper EUC 3605 * tables, and won't have copied any 3606 * EUC bytes. 3607 */ 3608 tp->t_col += bytes_moved; 3609 ibp->b_rptr += bytes_moved; 3610 obp->b_wptr += bytes_moved; 3611 bytes_left -= bytes_moved; 3612 if (ibp->b_rptr >= ibp->b_wptr) 3613 continue; /* moved all of block */ 3614 if (bytes_left == 0) { 3615 /* LINTED */ 3616 NEW_BLOCK(0); 3617 } 3618 c = *ibp->b_rptr++; /* stopper */ 3619 } 3620 3621 /* 3622 * Again, we need to make sure that this is not 3623 * a following byte of a multibyte character at 3624 * here. 3625 * 3626 * 'tp->t_eucign' will be 0 iff the current 'c' is 3627 * an ASCII character. Otherwise, it will have 3628 * the byte length of a multibyte character. 3629 * We also add the display width to 'tp->t_col' if 3630 * the current codeset is not UTF-8 since this is 3631 * a leading byte of a multibyte character. 3632 * For UTF-8 codeset type, we add the display width 3633 * when we get the last byte of a character. 3634 */ 3635 if ((tp->t_state & TS_MEUC) && tp->t_eucign == 0 && 3636 NOTASCII(c)) { 3637 tp->t_eucign = tp->t_csmethods.ldterm_memwidth( 3638 c, (void *)tp); 3639 tp->t_scratch_len = tp->t_eucign; 3640 3641 if (tp->t_csdata.codeset_type != 3642 LDTERM_CS_TYPE_UTF8) { 3643 tp->t_col += 3644 tp->t_csmethods.ldterm_dispwidth(c, 3645 (void *)tp, 3646 tp->t_modes.c_lflag & ECHOCTL); 3647 } 3648 } 3649 3650 /* 3651 * If the driver has requested, don't process 3652 * output flags. However, if we're in 3653 * multi-byte mode, we HAVE to look at 3654 * EVERYTHING going out to maintain column 3655 * position properly. Therefore IF the driver 3656 * says don't AND we're not doing multi-byte, 3657 * then don't do it. Otherwise, do it. 3658 * 3659 * NOTE: Hardware USUALLY doesn't expand tabs 3660 * properly for multi-byte situations anyway; 3661 * that's a known problem with the 3B2 3662 * "PORTS" board firmware, and any other 3663 * hardware that doesn't ACTUALLY know about 3664 * the current EUC mapping that WE are using 3665 * at this very moment. The problem is that 3666 * memory width is INDEPENDENT of screen 3667 * width - no relation - so WE know how wide 3668 * the characters are, but an off-the-host 3669 * board probably doesn't. So, until we're 3670 * SURE that the hardware below us can 3671 * correctly expand tabs in a 3672 * multi-byte/multi-column EUC situation, we 3673 * do it ourselves. 3674 */ 3675 /* 3676 * Map <CR>to<NL> on output if OCRNL flag 3677 * set. ONLCR processing is not done if OCRNL 3678 * is set. 3679 */ 3680 if (c == '\r' && (tp->t_modes.c_oflag & OCRNL)) { 3681 c = '\n'; 3682 ctype = typetab[c]; 3683 goto jocrnl; 3684 } 3685 3686 if (tp->t_csdata.codeset_type == LDTERM_CS_TYPE_EUC) { 3687 ctype = typetab[c]; 3688 } else { 3689 /* 3690 * In other codeset types, we safely assume 3691 * any byte of a multibyte character will have 3692 * 'ORDINARY' type. For ASCII characters, we 3693 * still use the typetab[]. 3694 */ 3695 if (tp->t_eucign == 0) 3696 ctype = typetab[c]; 3697 else 3698 ctype = ORDINARY; 3699 } 3700 3701 /* 3702 * Map <NL> to <CR><NL> on output if ONLCR 3703 * flag is set. 3704 */ 3705 if (c == '\n' && (tp->t_modes.c_oflag & ONLCR)) { 3706 if (!(tp->t_state & TS_TTCR)) { 3707 tp->t_state |= TS_TTCR; 3708 c = '\r'; 3709 ctype = typetab['\r']; 3710 --ibp->b_rptr; 3711 } else 3712 tp->t_state &= ~TS_TTCR; 3713 } 3714 /* 3715 * Delay values and column position 3716 * calculated here. For EUC chars in 3717 * multi-byte mode, we use "t_eucign" to help 3718 * calculate columns. When we see the first 3719 * byte of an EUC, we set t_eucign to the 3720 * number of bytes that will FOLLOW it, and 3721 * we add the screen width of the WHOLE EUC 3722 * character to the column position. In 3723 * particular, we can't count SS2 or SS3 as 3724 * printing characters. Remember, folks, the 3725 * screen width and memory width are 3726 * independent - no relation. We could have 3727 * dropped through for ASCII, but we want to 3728 * catch any bad characters (i.e., t_eucign 3729 * set and an ASCII char received) and 3730 * possibly report the garbage situation. 3731 */ 3732 jocrnl: 3733 3734 count = 0; 3735 switch (ctype) { 3736 3737 case T_SS2: 3738 case T_SS3: 3739 case ORDINARY: 3740 if (tp->t_state & TS_MEUC) { 3741 if (tp->t_eucign) { 3742 *obp->b_wptr++ = c; 3743 bytes_left--; 3744 3745 tp->t_scratch[tp->t_scratch_len 3746 - tp->t_eucign] = c; 3747 3748 --tp->t_eucign; 3749 3750 if (tp->t_csdata.codeset_type 3751 == LDTERM_CS_TYPE_UTF8 && 3752 tp->t_eucign <= 0) { 3753 tp->t_col += 3754 ldterm_utf8_width( 3755 tp->t_scratch, 3756 tp->t_scratch_len); 3757 } 3758 } else { 3759 if (tp->t_modes.c_oflag & OLCUC) 3760 n = elcuctab[c]; 3761 else 3762 n = enotrantab[c]; 3763 if (n) 3764 c = n; 3765 tp->t_col++; 3766 *obp->b_wptr++ = c; 3767 bytes_left--; 3768 } 3769 } else { /* ho hum, ASCII mode... */ 3770 if (tp->t_modes.c_oflag & OLCUC) 3771 n = lcuctab[c]; 3772 else 3773 n = notrantab[c]; 3774 if (n) 3775 c = n; 3776 tp->t_col++; 3777 *obp->b_wptr++ = c; 3778 bytes_left--; 3779 } 3780 break; 3781 3782 /* 3783 * If we're doing ECHOCTL, we've 3784 * already mapped the thing during 3785 * the process of canonising. Don't 3786 * bother here, as it's not one that 3787 * we did. 3788 */ 3789 case CONTROL: 3790 *obp->b_wptr++ = c; 3791 bytes_left--; 3792 break; 3793 3794 /* 3795 * This is probably a backspace 3796 * received, not one that we're 3797 * echoing. Let it go as a 3798 * single-column backspace. 3799 */ 3800 case BACKSPACE: 3801 if (tp->t_col) 3802 tp->t_col--; 3803 if (tp->t_modes.c_oflag & BSDLY) { 3804 if (tp->t_modes.c_oflag & OFILL) 3805 count = 1; 3806 } 3807 *obp->b_wptr++ = c; 3808 bytes_left--; 3809 break; 3810 3811 case NEWLINE: 3812 if (tp->t_modes.c_oflag & ONLRET) 3813 goto cr; 3814 if ((tp->t_modes.c_oflag & NLDLY) == NL1) 3815 count = 2; 3816 *obp->b_wptr++ = c; 3817 bytes_left--; 3818 break; 3819 3820 case TAB: 3821 /* 3822 * Map '\t' to spaces if XTABS flag 3823 * is set. The calculation of 3824 * "t_eucign" has probably insured 3825 * that column will be correct, as we 3826 * bumped t_col by the DISP width, 3827 * not the memory width. 3828 */ 3829 if ((tp->t_modes.c_oflag & TABDLY) == XTABS) { 3830 for (;;) { 3831 *obp->b_wptr++ = ' '; 3832 bytes_left--; 3833 tp->t_col++; 3834 if ((tp->t_col & 07) == 0) 3835 break; /* every 8th */ 3836 /* 3837 * If we don't have 3838 * room to fully 3839 * expand this tab in 3840 * this block, back 3841 * up to continue 3842 * expanding it into 3843 * the next block. 3844 */ 3845 if (obp->b_wptr >= 3846 obp->b_datap->db_lim) { 3847 ibp->b_rptr--; 3848 break; 3849 } 3850 } 3851 } else { 3852 tp->t_col |= 07; 3853 tp->t_col++; 3854 if (tp->t_modes.c_oflag & OFILL) { 3855 if (tp->t_modes.c_oflag & 3856 TABDLY) 3857 count = 2; 3858 } else { 3859 switch (tp->t_modes.c_oflag & 3860 TABDLY) { 3861 case TAB2: 3862 count = 6; 3863 break; 3864 3865 case TAB1: 3866 count = 1 + (tp->t_col | 3867 ~07); 3868 if (count < 5) 3869 count = 0; 3870 break; 3871 } 3872 } 3873 *obp->b_wptr++ = c; 3874 bytes_left--; 3875 } 3876 break; 3877 3878 case VTAB: 3879 if ((tp->t_modes.c_oflag & VTDLY) && 3880 !(tp->t_modes.c_oflag & OFILL)) 3881 count = 127; 3882 *obp->b_wptr++ = c; 3883 bytes_left--; 3884 break; 3885 3886 case RETURN: 3887 /* 3888 * Ignore <CR> in column 0 if ONOCR 3889 * flag set. 3890 */ 3891 if (tp->t_col == 0 && 3892 (tp->t_modes.c_oflag & ONOCR)) 3893 break; 3894 3895 cr: 3896 switch (tp->t_modes.c_oflag & CRDLY) { 3897 3898 case CR1: 3899 if (tp->t_modes.c_oflag & OFILL) 3900 count = 2; 3901 else 3902 count = tp->t_col % 2; 3903 break; 3904 3905 case CR2: 3906 if (tp->t_modes.c_oflag & OFILL) 3907 count = 4; 3908 else 3909 count = 6; 3910 break; 3911 3912 case CR3: 3913 if (tp->t_modes.c_oflag & OFILL) 3914 count = 0; 3915 else 3916 count = 9; 3917 break; 3918 } 3919 tp->t_col = 0; 3920 *obp->b_wptr++ = c; 3921 bytes_left--; 3922 break; 3923 } 3924 3925 if (count != 0) { 3926 if (tp->t_modes.c_oflag & OFILL) { 3927 do { 3928 if (bytes_left == 0) { 3929 /* LINTED */ 3930 NEW_BLOCK(0); 3931 } 3932 if (tp->t_modes.c_oflag & OFDEL) 3933 *obp->b_wptr++ = CDEL; 3934 else 3935 *obp->b_wptr++ = CNUL; 3936 bytes_left--; 3937 } while (--count != 0); 3938 } else { 3939 if ((tp->t_modes.c_lflag & FLUSHO) && 3940 (tp->t_modes.c_lflag & IEXTEN)) { 3941 /* drop on floor */ 3942 freemsg(*omp); 3943 } else { 3944 /* 3945 * Update sysinfo 3946 * outch 3947 */ 3948 (void) drv_setparm(SYSOUTC, 3949 msgdsize(*omp)); 3950 putnext(q, *omp); 3951 /* 3952 * Send M_DELAY 3953 * downstream 3954 */ 3955 if ((bp = 3956 allocb(1, BPRI_MED)) != 3957 NULL) { 3958 bp->b_datap->db_type = 3959 M_DELAY; 3960 *bp->b_wptr++ = 3961 (uchar_t)count; 3962 putnext(q, bp); 3963 } 3964 } 3965 bytes_left = 0; 3966 /* 3967 * We have to start a new 3968 * message; the delay 3969 * introduces a break between 3970 * messages. 3971 */ 3972 *omp = NULL; 3973 contpp = omp; 3974 } 3975 } 3976 } 3977 cbp = ibp->b_cont; 3978 freeb(ibp); 3979 } while ((ibp = cbp) != NULL); /* next block, if any */ 3980 3981 outofbufs: 3982 return (ibp); 3983 #undef NEW_BLOCK 3984 } 3985 3986 3987 #if !defined(__sparc) 3988 int 3989 movtuc(size_t size, unsigned char *from, unsigned char *origto, 3990 unsigned char *table) 3991 { 3992 unsigned char *to = origto; 3993 unsigned char c; 3994 3995 while (size != 0 && (c = table[*from++]) != 0) { 3996 *to++ = c; 3997 size--; 3998 } 3999 return (to - origto); 4000 } 4001 #endif 4002 4003 static void 4004 ldterm_flush_output(uchar_t c, queue_t *q, ldtermstd_state_t *tp) 4005 { 4006 /* Already conditioned with IEXTEN during VDISCARD processing */ 4007 if (tp->t_modes.c_lflag & FLUSHO) 4008 tp->t_modes.c_lflag &= ~FLUSHO; 4009 else { 4010 flushq(q, FLUSHDATA); /* flush our write queue */ 4011 /* flush ones below us */ 4012 (void) putnextctl1(q, M_FLUSH, FLUSHW); 4013 if ((tp->t_echomp = allocb(EBSIZE, BPRI_HI)) != NULL) { 4014 (void) ldterm_echo(c, q, 1, tp); 4015 if (tp->t_msglen != 0) 4016 ldterm_reprint(q, EBSIZE, tp); 4017 if (tp->t_echomp != NULL) { 4018 putnext(q, tp->t_echomp); 4019 tp->t_echomp = NULL; 4020 } 4021 } 4022 tp->t_modes.c_lflag |= FLUSHO; 4023 } 4024 } 4025 4026 4027 /* 4028 * Signal generated by the reader: M_PCSIG and M_FLUSH messages sent. 4029 */ 4030 static void 4031 ldterm_dosig(queue_t *q, int sig, uchar_t c, int mtype, int mode) 4032 { 4033 ldtermstd_state_t *tp = (ldtermstd_state_t *)q->q_ptr; 4034 int sndsig = 0; 4035 4036 /* 4037 * c == \0 is brk case; need to flush on BRKINT even if 4038 * noflsh is set. 4039 */ 4040 if ((!(tp->t_modes.c_lflag & NOFLSH)) || (c == '\0')) { 4041 if (mode) { 4042 if (tp->t_state & TS_TTSTOP) { 4043 sndsig = 1; 4044 (void) putnextctl1(q, mtype, sig); 4045 } 4046 /* 4047 * Flush read or write side. 4048 * Restart the input or output. 4049 */ 4050 if (mode & FLUSHR) { 4051 flushq(q, FLUSHDATA); 4052 (void) putnextctl1(WR(q), M_FLUSH, mode); 4053 if (tp->t_state & (TS_TBLOCK|TS_IFBLOCK)) { 4054 (void) putnextctl(WR(q), M_STARTI); 4055 tp->t_state &= ~(TS_TBLOCK|TS_IFBLOCK); 4056 } 4057 } 4058 if (mode & FLUSHW) { 4059 flushq(WR(q), FLUSHDATA); 4060 /* 4061 * XXX This is extremely gross. 4062 * Since we can't be sure our M_FLUSH 4063 * will have run its course by the 4064 * time we do the echo below, we set 4065 * state and toss it in the write put 4066 * routine to prevent flushing our 4067 * own data. Note that downstream 4068 * modules on the write side will be 4069 * flushed by the M_FLUSH sent above. 4070 */ 4071 tp->t_state |= TS_FLUSHWAIT; 4072 (void) putnextctl1(q, M_FLUSH, FLUSHW); 4073 if (tp->t_state & TS_TTSTOP) { 4074 (void) putnextctl(WR(q), M_START); 4075 tp->t_state &= ~(TS_TTSTOP|TS_OFBLOCK); 4076 } 4077 } 4078 } 4079 } 4080 tp->t_state &= ~TS_QUOT; 4081 if (sndsig == 0) 4082 (void) putnextctl1(q, mtype, sig); 4083 4084 if (c != '\0') { 4085 if ((tp->t_echomp = allocb(4, BPRI_HI)) != NULL) { 4086 if (ldterm_echo(c, WR(q), 4, tp) > 0) 4087 putnext(WR(q), tp->t_echomp); 4088 else 4089 freemsg(tp->t_echomp); 4090 tp->t_echomp = NULL; 4091 } 4092 } 4093 } 4094 4095 4096 /* 4097 * Called when an M_IOCTL message is seen on the write queue; does 4098 * whatever we're supposed to do with it, and either replies 4099 * immediately or passes it to the next module down. 4100 */ 4101 static void 4102 ldterm_do_ioctl(queue_t *q, mblk_t *mp) 4103 { 4104 ldtermstd_state_t *tp; 4105 struct iocblk *iocp; 4106 struct eucioc *euciocp; /* needed for EUC ioctls */ 4107 ldterm_cs_data_user_t *csdp; 4108 int i; 4109 int locale_name_sz; 4110 uchar_t maxbytelen; 4111 uchar_t maxscreenlen; 4112 int error; 4113 4114 iocp = (struct iocblk *)mp->b_rptr; 4115 tp = (ldtermstd_state_t *)q->q_ptr; 4116 4117 switch (iocp->ioc_cmd) { 4118 4119 case TCSETS: 4120 case TCSETSW: 4121 case TCSETSF: 4122 { 4123 /* 4124 * Set current parameters and special 4125 * characters. 4126 */ 4127 struct termios *cb; 4128 struct termios oldmodes; 4129 4130 error = miocpullup(mp, sizeof (struct termios)); 4131 if (error != 0) { 4132 miocnak(q, mp, 0, error); 4133 return; 4134 } 4135 4136 cb = (struct termios *)mp->b_cont->b_rptr; 4137 4138 oldmodes = tp->t_amodes; 4139 tp->t_amodes = *cb; 4140 if ((tp->t_amodes.c_lflag & PENDIN) && 4141 (tp->t_modes.c_lflag & IEXTEN)) { 4142 /* 4143 * Yuk. The C shell file completion 4144 * code actually uses this "feature", 4145 * so we have to support it. 4146 */ 4147 if (tp->t_message != NULL) { 4148 tp->t_state |= TS_RESCAN; 4149 qenable(RD(q)); 4150 } 4151 tp->t_amodes.c_lflag &= ~PENDIN; 4152 } 4153 bcopy(tp->t_amodes.c_cc, tp->t_modes.c_cc, NCCS); 4154 4155 /* 4156 * ldterm_adjust_modes does not deal with 4157 * cflags 4158 */ 4159 tp->t_modes.c_cflag = tp->t_amodes.c_cflag; 4160 4161 ldterm_adjust_modes(tp); 4162 if (chgstropts(&oldmodes, tp, RD(q)) == (-1)) { 4163 miocnak(q, mp, 0, EAGAIN); 4164 return; 4165 } 4166 /* 4167 * The driver may want to know about the 4168 * following iflags: IGNBRK, BRKINT, IGNPAR, 4169 * PARMRK, INPCK, IXON, IXANY. 4170 */ 4171 break; 4172 } 4173 4174 case TCSETA: 4175 case TCSETAW: 4176 case TCSETAF: 4177 { 4178 /* 4179 * Old-style "ioctl" to set current 4180 * parameters and special characters. Don't 4181 * clear out the unset portions, leave them 4182 * as they are. 4183 */ 4184 struct termio *cb; 4185 struct termios oldmodes; 4186 4187 error = miocpullup(mp, sizeof (struct termio)); 4188 if (error != 0) { 4189 miocnak(q, mp, 0, error); 4190 return; 4191 } 4192 4193 cb = (struct termio *)mp->b_cont->b_rptr; 4194 4195 oldmodes = tp->t_amodes; 4196 tp->t_amodes.c_iflag = 4197 (tp->t_amodes.c_iflag & 0xffff0000 | cb->c_iflag); 4198 tp->t_amodes.c_oflag = 4199 (tp->t_amodes.c_oflag & 0xffff0000 | cb->c_oflag); 4200 tp->t_amodes.c_cflag = 4201 (tp->t_amodes.c_cflag & 0xffff0000 | cb->c_cflag); 4202 tp->t_amodes.c_lflag = 4203 (tp->t_amodes.c_lflag & 0xffff0000 | cb->c_lflag); 4204 4205 bcopy(cb->c_cc, tp->t_modes.c_cc, NCC); 4206 /* TCGETS returns amodes, so update that too */ 4207 bcopy(cb->c_cc, tp->t_amodes.c_cc, NCC); 4208 4209 /* ldterm_adjust_modes does not deal with cflags */ 4210 4211 tp->t_modes.c_cflag = tp->t_amodes.c_cflag; 4212 4213 ldterm_adjust_modes(tp); 4214 if (chgstropts(&oldmodes, tp, RD(q)) == (-1)) { 4215 miocnak(q, mp, 0, EAGAIN); 4216 return; 4217 } 4218 /* 4219 * The driver may want to know about the 4220 * following iflags: IGNBRK, BRKINT, IGNPAR, 4221 * PARMRK, INPCK, IXON, IXANY. 4222 */ 4223 break; 4224 } 4225 4226 case TCFLSH: 4227 /* 4228 * Do the flush on the write queue immediately, and 4229 * queue up any flush on the read queue for the 4230 * service procedure to see. Then turn it into the 4231 * appropriate M_FLUSH message, so that the module 4232 * below us doesn't have to know about TCFLSH. 4233 */ 4234 error = miocpullup(mp, sizeof (int)); 4235 if (error != 0) { 4236 miocnak(q, mp, 0, error); 4237 return; 4238 } 4239 4240 ASSERT(mp->b_datap != NULL); 4241 if (*(int *)mp->b_cont->b_rptr == 0) { 4242 ASSERT(mp->b_datap != NULL); 4243 (void) putnextctl1(q, M_FLUSH, FLUSHR); 4244 (void) putctl1(RD(q), M_FLUSH, FLUSHR); 4245 } else if (*(int *)mp->b_cont->b_rptr == 1) { 4246 flushq(q, FLUSHDATA); 4247 ASSERT(mp->b_datap != NULL); 4248 tp->t_state |= TS_FLUSHWAIT; 4249 (void) putnextctl1(RD(q), M_FLUSH, FLUSHW); 4250 (void) putnextctl1(q, M_FLUSH, FLUSHW); 4251 } else if (*(int *)mp->b_cont->b_rptr == 2) { 4252 flushq(q, FLUSHDATA); 4253 ASSERT(mp->b_datap != NULL); 4254 (void) putnextctl1(q, M_FLUSH, FLUSHRW); 4255 tp->t_state |= TS_FLUSHWAIT; 4256 (void) putnextctl1(RD(q), M_FLUSH, FLUSHRW); 4257 } else { 4258 miocnak(q, mp, 0, EINVAL); 4259 return; 4260 } 4261 ASSERT(mp->b_datap != NULL); 4262 iocp->ioc_rval = 0; 4263 miocack(q, mp, 0, 0); 4264 return; 4265 4266 case TCXONC: 4267 error = miocpullup(mp, sizeof (int)); 4268 if (error != 0) { 4269 miocnak(q, mp, 0, error); 4270 return; 4271 } 4272 4273 switch (*(int *)mp->b_cont->b_rptr) { 4274 case 0: 4275 if (!(tp->t_state & TS_TTSTOP)) { 4276 (void) putnextctl(q, M_STOP); 4277 tp->t_state |= (TS_TTSTOP|TS_OFBLOCK); 4278 } 4279 break; 4280 4281 case 1: 4282 if (tp->t_state & TS_TTSTOP) { 4283 (void) putnextctl(q, M_START); 4284 tp->t_state &= ~(TS_TTSTOP|TS_OFBLOCK); 4285 } 4286 break; 4287 4288 case 2: 4289 (void) putnextctl(q, M_STOPI); 4290 tp->t_state |= (TS_TBLOCK|TS_IFBLOCK); 4291 break; 4292 4293 case 3: 4294 (void) putnextctl(q, M_STARTI); 4295 tp->t_state &= ~(TS_TBLOCK|TS_IFBLOCK); 4296 break; 4297 4298 default: 4299 miocnak(q, mp, 0, EINVAL); 4300 return; 4301 } 4302 ASSERT(mp->b_datap != NULL); 4303 iocp->ioc_rval = 0; 4304 miocack(q, mp, 0, 0); 4305 return; 4306 /* 4307 * TCSBRK is expected to be handled by the driver. 4308 * The reason its left for the driver is that when 4309 * the argument to TCSBRK is zero driver has to drain 4310 * the data and sending a M_IOCACK from LDTERM before 4311 * the driver drains the data is going to cause 4312 * problems. 4313 */ 4314 4315 /* 4316 * The following are EUC related ioctls. For 4317 * EUC_WSET, we have to pass the information on, even 4318 * though we ACK the call. It's vital in the EUC 4319 * environment that everybody downstream knows about 4320 * the EUC codeset widths currently in use; we 4321 * therefore pass down the information in an M_CTL 4322 * message. It will bottom out in the driver. 4323 */ 4324 case EUC_WSET: 4325 { 4326 4327 /* only needed for EUC_WSET */ 4328 struct iocblk *riocp; 4329 4330 mblk_t *dmp, *dmp_cont; 4331 4332 /* 4333 * If the user didn't supply any information, 4334 * NAK it. 4335 */ 4336 error = miocpullup(mp, sizeof (struct eucioc)); 4337 if (error != 0) { 4338 miocnak(q, mp, 0, error); 4339 return; 4340 } 4341 4342 euciocp = (struct eucioc *)mp->b_cont->b_rptr; 4343 /* 4344 * Check here for something reasonable. If 4345 * anything will take more than EUC_MAXW 4346 * columns or more than EUC_MAXW bytes 4347 * following SS2 or SS3, then just reject it 4348 * out of hand. It's not impossible for us to 4349 * do it, it just isn't reasonable. So far, 4350 * in the world, we've seen the absolute max 4351 * columns to be 2 and the max number of 4352 * bytes to be 3. This allows room for some 4353 * expansion of that, but it probably won't 4354 * even be necessary. At the moment, we 4355 * return a "range" error. If you really 4356 * need to, you can push EUC_MAXW up to over 4357 * 200; it doesn't make sense, though, with 4358 * only a CANBSIZ sized input limit (usually 4359 * 256)! 4360 */ 4361 for (i = 0; i < 4; i++) { 4362 if ((euciocp->eucw[i] > EUC_MAXW) || 4363 (euciocp->scrw[i] > EUC_MAXW)) { 4364 miocnak(q, mp, 0, ERANGE); 4365 return; 4366 } 4367 } 4368 /* 4369 * Otherwise, save the information in tp, 4370 * force codeset 0 (ASCII) to be one byte, 4371 * one column. 4372 */ 4373 cp_eucwioc(euciocp, &tp->eucwioc, EUCIN); 4374 tp->eucwioc.eucw[0] = tp->eucwioc.scrw[0] = 1; 4375 /* 4376 * Now, check out whether we're doing 4377 * multibyte processing. if we are, we need 4378 * to allocate a block to hold the parallel 4379 * array. By convention, we've been passed 4380 * what amounts to a CSWIDTH definition. We 4381 * actually NEED the number of bytes for 4382 * Codesets 2 & 3. 4383 */ 4384 tp->t_maxeuc = 0; /* reset to say we're NOT */ 4385 4386 tp->t_state &= ~TS_MEUC; 4387 /* 4388 * We'll set TS_MEUC if we're doing 4389 * multi-column OR multi- byte OR both. It 4390 * makes things easier... NOTE: If we fail 4391 * to get the buffer we need to hold display 4392 * widths, then DON'T let the TS_MEUC bit get 4393 * set! 4394 */ 4395 for (i = 0; i < 4; i++) { 4396 if (tp->eucwioc.eucw[i] > tp->t_maxeuc) 4397 tp->t_maxeuc = tp->eucwioc.eucw[i]; 4398 if (tp->eucwioc.scrw[i] > 1) 4399 tp->t_state |= TS_MEUC; 4400 } 4401 if ((tp->t_maxeuc > 1) || (tp->t_state & TS_MEUC)) { 4402 if (!tp->t_eucp_mp) { 4403 if ((tp->t_eucp_mp = allocb(_TTY_BUFSIZ, 4404 BPRI_HI)) == NULL) { 4405 tp->t_maxeuc = 1; 4406 tp->t_state &= ~TS_MEUC; 4407 cmn_err(CE_WARN, 4408 "Can't allocate eucp_mp"); 4409 miocnak(q, mp, 0, ENOSR); 4410 return; 4411 } 4412 /* 4413 * here, if there's junk in 4414 * the canonical buffer, then 4415 * move the eucp pointer past 4416 * it, so we don't run off 4417 * the beginning. This is a 4418 * total botch, but will 4419 * hopefully keep stuff from 4420 * getting too messed up 4421 * until the user flushes 4422 * this line! 4423 */ 4424 if (tp->t_msglen) { 4425 tp->t_eucp = 4426 tp->t_eucp_mp->b_rptr; 4427 for (i = tp->t_msglen; i; i--) 4428 *tp->t_eucp++ = 1; 4429 } else { 4430 tp->t_eucp = 4431 tp->t_eucp_mp->b_rptr; 4432 } 4433 } 4434 /* doing multi-byte handling */ 4435 tp->t_state |= TS_MEUC; 4436 4437 } else if (tp->t_eucp_mp) { 4438 freemsg(tp->t_eucp_mp); 4439 tp->t_eucp_mp = NULL; 4440 tp->t_eucp = NULL; 4441 } 4442 4443 /* 4444 * Save the EUC width data we have at 4445 * the t_csdata, set t_csdata.codeset_type to 4446 * EUC one, and, switch the codeset methods at 4447 * t_csmethods. 4448 */ 4449 bzero(&tp->t_csdata.eucpc_data, 4450 (sizeof (ldterm_eucpc_data_t) * 4451 LDTERM_CS_MAX_CODESETS)); 4452 tp->t_csdata.eucpc_data[0].byte_length = 4453 tp->eucwioc.eucw[1]; 4454 tp->t_csdata.eucpc_data[0].screen_width = 4455 tp->eucwioc.scrw[1]; 4456 tp->t_csdata.eucpc_data[1].byte_length = 4457 tp->eucwioc.eucw[2]; 4458 tp->t_csdata.eucpc_data[1].screen_width = 4459 tp->eucwioc.scrw[2]; 4460 tp->t_csdata.eucpc_data[2].byte_length = 4461 tp->eucwioc.eucw[3]; 4462 tp->t_csdata.eucpc_data[2].screen_width = 4463 tp->eucwioc.scrw[3]; 4464 tp->t_csdata.version = LDTERM_DATA_VERSION; 4465 tp->t_csdata.codeset_type = LDTERM_CS_TYPE_EUC; 4466 /* 4467 * We are not using the 'csinfo_num' anyway if the 4468 * current codeset type is EUC. So, set it to 4469 * the maximum possible. 4470 */ 4471 tp->t_csdata.csinfo_num = 4472 LDTERM_CS_TYPE_EUC_MAX_SUBCS; 4473 if (tp->t_csdata.locale_name != (char *)NULL) { 4474 kmem_free(tp->t_csdata.locale_name, 4475 strlen(tp->t_csdata.locale_name) + 1); 4476 tp->t_csdata.locale_name = (char *)NULL; 4477 } 4478 tp->t_csmethods = cs_methods[LDTERM_CS_TYPE_EUC]; 4479 4480 /* 4481 * If we are able to allocate two blocks (the 4482 * iocblk and the associated data), then pass 4483 * it downstream, otherwise we'll need to NAK 4484 * it, and drop whatever we WERE able to 4485 * allocate. 4486 */ 4487 if ((dmp = mkiocb(EUC_WSET)) == NULL) { 4488 miocnak(q, mp, 0, ENOSR); 4489 return; 4490 } 4491 if ((dmp_cont = allocb(EUCSIZE, BPRI_HI)) == NULL) { 4492 freemsg(dmp); 4493 miocnak(q, mp, 0, ENOSR); 4494 return; 4495 } 4496 4497 /* 4498 * We got both buffers. Copy out the EUC 4499 * information (as we received it, not what 4500 * we're using!) & pass it on. 4501 */ 4502 bcopy(mp->b_cont->b_rptr, dmp_cont->b_rptr, EUCSIZE); 4503 dmp_cont->b_wptr += EUCSIZE; 4504 dmp->b_cont = dmp_cont; 4505 dmp->b_datap->db_type = M_CTL; 4506 dmp_cont->b_datap->db_type = M_DATA; 4507 riocp = (struct iocblk *)dmp->b_rptr; 4508 riocp->ioc_count = EUCSIZE; 4509 putnext(q, dmp); 4510 4511 /* 4512 * Now ACK the ioctl. 4513 */ 4514 iocp->ioc_rval = 0; 4515 miocack(q, mp, 0, 0); 4516 return; 4517 } 4518 4519 case EUC_WGET: 4520 error = miocpullup(mp, sizeof (struct eucioc)); 4521 if (error != 0) { 4522 miocnak(q, mp, 0, error); 4523 return; 4524 } 4525 euciocp = (struct eucioc *)mp->b_cont->b_rptr; 4526 cp_eucwioc(&tp->eucwioc, euciocp, EUCOUT); 4527 iocp->ioc_rval = 0; 4528 miocack(q, mp, EUCSIZE, 0); 4529 return; 4530 4531 case CSDATA_SET: 4532 error = miocpullup(mp, sizeof (ldterm_cs_data_user_t)); 4533 if (error != 0) { 4534 miocnak(q, mp, 0, error); 4535 return; 4536 } 4537 4538 csdp = (ldterm_cs_data_user_t *)mp->b_cont->b_rptr; 4539 4540 /* Validate the codeset data provided. */ 4541 if (csdp->version > LDTERM_DATA_VERSION || 4542 csdp->codeset_type < LDTERM_CS_TYPE_MIN || 4543 csdp->codeset_type > LDTERM_CS_TYPE_MAX) { 4544 miocnak(q, mp, 0, ERANGE); 4545 return; 4546 } 4547 4548 if ((csdp->codeset_type == LDTERM_CS_TYPE_EUC && 4549 csdp->csinfo_num > LDTERM_CS_TYPE_EUC_MAX_SUBCS) || 4550 (csdp->codeset_type == LDTERM_CS_TYPE_PCCS && 4551 (csdp->csinfo_num < LDTERM_CS_TYPE_PCCS_MIN_SUBCS || 4552 csdp->csinfo_num > LDTERM_CS_TYPE_PCCS_MAX_SUBCS))) { 4553 miocnak(q, mp, 0, ERANGE); 4554 return; 4555 } 4556 4557 maxbytelen = maxscreenlen = 0; 4558 if (csdp->codeset_type == LDTERM_CS_TYPE_EUC) { 4559 for (i = 0; i < LDTERM_CS_TYPE_EUC_MAX_SUBCS; i++) { 4560 if (csdp->eucpc_data[i].byte_length > 4561 EUC_MAXW || 4562 csdp->eucpc_data[i].screen_width > 4563 EUC_MAXW) { 4564 miocnak(q, mp, 0, ERANGE); 4565 return; 4566 } 4567 4568 if (csdp->eucpc_data[i].byte_length > 4569 maxbytelen) 4570 maxbytelen = 4571 csdp->eucpc_data[i].byte_length; 4572 if (csdp->eucpc_data[i].screen_width > 4573 maxscreenlen) 4574 maxscreenlen = 4575 csdp->eucpc_data[i].screen_width; 4576 } 4577 /* POSIX/C locale? */ 4578 if (maxbytelen == 0 && maxscreenlen == 0) 4579 maxbytelen = maxscreenlen = 1; 4580 } else if (csdp->codeset_type == LDTERM_CS_TYPE_PCCS) { 4581 for (i = 0; i < LDTERM_CS_MAX_CODESETS; i++) { 4582 if (csdp->eucpc_data[i].byte_length > 4583 LDTERM_CS_MAX_BYTE_LENGTH) { 4584 miocnak(q, mp, 0, ERANGE); 4585 return; 4586 } 4587 if (csdp->eucpc_data[i].byte_length > 4588 maxbytelen) 4589 maxbytelen = 4590 csdp->eucpc_data[i].byte_length; 4591 if (csdp->eucpc_data[i].screen_width > 4592 maxscreenlen) 4593 maxscreenlen = 4594 csdp->eucpc_data[i].screen_width; 4595 } 4596 } else if (csdp->codeset_type == LDTERM_CS_TYPE_UTF8) { 4597 maxbytelen = 4; 4598 maxscreenlen = 2; 4599 } 4600 4601 locale_name_sz = 0; 4602 if (csdp->locale_name) { 4603 for (i = 0; i < MAXNAMELEN; i++) 4604 if (csdp->locale_name[i] == '\0') 4605 break; 4606 /* 4607 * We cannot have any string that is not NULL byte 4608 * terminated. 4609 */ 4610 if (i >= MAXNAMELEN) { 4611 miocnak(q, mp, 0, ERANGE); 4612 return; 4613 } 4614 4615 locale_name_sz = i + 1; 4616 } 4617 4618 /* 4619 * As the final check, if there was invalid codeset_type 4620 * given, or invalid byte_length was specified, it's an error. 4621 */ 4622 if (maxbytelen <= 0 || maxscreenlen <= 0) { 4623 miocnak(q, mp, 0, ERANGE); 4624 return; 4625 } 4626 4627 /* Do the switching. */ 4628 tp->t_maxeuc = maxbytelen; 4629 tp->t_state &= ~TS_MEUC; 4630 if (maxbytelen > 1 || maxscreenlen > 1) { 4631 if (!tp->t_eucp_mp) { 4632 if (!(tp->t_eucp_mp = allocb(_TTY_BUFSIZ, 4633 BPRI_HI))) { 4634 cmn_err(CE_WARN, 4635 "Can't allocate eucp_mp"); 4636 miocnak(q, mp, 0, ENOSR); 4637 return; 4638 } 4639 /* 4640 * If there's junk in the canonical buffer, 4641 * then move the eucp pointer past it, 4642 * so we don't run off the beginning. This is 4643 * a total botch, but will hopefully keep 4644 * stuff from getting too messed up until 4645 * the user flushes this line! 4646 */ 4647 if (tp->t_msglen) { 4648 tp->t_eucp = tp->t_eucp_mp->b_rptr; 4649 for (i = tp->t_msglen; i; i--) 4650 *tp->t_eucp++ = 1; 4651 } else { 4652 tp->t_eucp = tp->t_eucp_mp->b_rptr; 4653 } 4654 } 4655 4656 /* 4657 * We only set TS_MEUC for a multibyte/multi-column 4658 * codeset. 4659 */ 4660 tp->t_state |= TS_MEUC; 4661 4662 tp->t_csdata.version = csdp->version; 4663 tp->t_csdata.codeset_type = csdp->codeset_type; 4664 tp->t_csdata.csinfo_num = csdp->csinfo_num; 4665 bcopy(csdp->eucpc_data, tp->t_csdata.eucpc_data, 4666 sizeof (ldterm_eucpc_data_t) * 4667 LDTERM_CS_MAX_CODESETS); 4668 tp->t_csmethods = cs_methods[csdp->codeset_type]; 4669 4670 if (csdp->codeset_type == LDTERM_CS_TYPE_EUC) { 4671 tp->eucwioc.eucw[0] = 1; 4672 tp->eucwioc.scrw[0] = 1; 4673 4674 tp->eucwioc.eucw[1] = 4675 csdp->eucpc_data[0].byte_length; 4676 tp->eucwioc.scrw[1] = 4677 csdp->eucpc_data[0].screen_width; 4678 4679 tp->eucwioc.eucw[2] = 4680 csdp->eucpc_data[1].byte_length + 1; 4681 tp->eucwioc.scrw[2] = 4682 csdp->eucpc_data[1].screen_width; 4683 4684 tp->eucwioc.eucw[3] = 4685 csdp->eucpc_data[2].byte_length + 1; 4686 tp->eucwioc.scrw[3] = 4687 csdp->eucpc_data[2].screen_width; 4688 } else { 4689 /* 4690 * We are not going to use this data 4691 * structure. So, clear it. Also, stty(1) will 4692 * make use of the cleared tp->eucwioc when 4693 * it prints out codeset width setting. 4694 */ 4695 bzero(&tp->eucwioc, EUCSIZE); 4696 } 4697 } else { 4698 /* 4699 * If this codeset is a single byte codeset that 4700 * requires only single display column for all 4701 * characters, we switch to default EUC codeset 4702 * methods and data setting. 4703 */ 4704 4705 if (tp->t_eucp_mp) { 4706 freemsg(tp->t_eucp_mp); 4707 tp->t_eucp_mp = NULL; 4708 tp->t_eucp = NULL; 4709 } 4710 4711 bzero(&tp->eucwioc, EUCSIZE); 4712 tp->eucwioc.eucw[0] = 1; 4713 tp->eucwioc.scrw[0] = 1; 4714 if (tp->t_csdata.locale_name != (char *)NULL) { 4715 kmem_free(tp->t_csdata.locale_name, 4716 strlen(tp->t_csdata.locale_name) + 1); 4717 } 4718 tp->t_csdata = default_cs_data; 4719 tp->t_csmethods = cs_methods[LDTERM_CS_TYPE_EUC]; 4720 } 4721 4722 /* Copy over locale_name. */ 4723 if (tp->t_csdata.locale_name != (char *)NULL) { 4724 kmem_free(tp->t_csdata.locale_name, 4725 strlen(tp->t_csdata.locale_name) + 1); 4726 } 4727 if (locale_name_sz > 1) { 4728 tp->t_csdata.locale_name = (char *)kmem_alloc( 4729 locale_name_sz, KM_SLEEP); 4730 (void) strcpy(tp->t_csdata.locale_name, 4731 csdp->locale_name); 4732 } else { 4733 tp->t_csdata.locale_name = (char *)NULL; 4734 } 4735 4736 /* 4737 * Now ACK the ioctl. 4738 */ 4739 iocp->ioc_rval = 0; 4740 miocack(q, mp, 0, 0); 4741 return; 4742 4743 case CSDATA_GET: 4744 error = miocpullup(mp, sizeof (ldterm_cs_data_user_t)); 4745 if (error != 0) { 4746 miocnak(q, mp, 0, error); 4747 return; 4748 } 4749 4750 csdp = (ldterm_cs_data_user_t *)mp->b_cont->b_rptr; 4751 4752 csdp->version = tp->t_csdata.version; 4753 csdp->codeset_type = tp->t_csdata.codeset_type; 4754 csdp->csinfo_num = tp->t_csdata.csinfo_num; 4755 csdp->pad = tp->t_csdata.pad; 4756 if (tp->t_csdata.locale_name) { 4757 (void) strcpy(csdp->locale_name, 4758 tp->t_csdata.locale_name); 4759 } else { 4760 csdp->locale_name[0] = '\0'; 4761 } 4762 bcopy(tp->t_csdata.eucpc_data, csdp->eucpc_data, 4763 sizeof (ldterm_eucpc_data_t) * LDTERM_CS_MAX_CODESETS); 4764 /* 4765 * If the codeset is an EUC codeset and if it has 2nd and/or 4766 * 3rd supplementary codesets, we subtract one from each 4767 * byte length of the supplementary codesets. This is 4768 * because single shift characters, SS2 and SS3, are not 4769 * included in the byte lengths in the user space. 4770 */ 4771 if (csdp->codeset_type == LDTERM_CS_TYPE_EUC) { 4772 if (csdp->eucpc_data[1].byte_length) 4773 csdp->eucpc_data[1].byte_length -= 1; 4774 if (csdp->eucpc_data[2].byte_length) 4775 csdp->eucpc_data[2].byte_length -= 1; 4776 } 4777 iocp->ioc_rval = 0; 4778 miocack(q, mp, sizeof (ldterm_cs_data_user_t), 0); 4779 return; 4780 4781 case PTSSTTY: 4782 tp->t_state |= TS_ISPTSTTY; 4783 break; 4784 4785 } 4786 4787 putnext(q, mp); 4788 } 4789 4790 4791 /* 4792 * Send an M_SETOPTS message upstream if any mode changes are being 4793 * made that affect the stream head options. returns -1 if allocb 4794 * fails, else returns 0. 4795 */ 4796 static int 4797 chgstropts(struct termios *oldmodep, ldtermstd_state_t *tp, queue_t *q) 4798 { 4799 struct stroptions optbuf; 4800 mblk_t *bp; 4801 4802 optbuf.so_flags = 0; 4803 if ((oldmodep->c_lflag ^ tp->t_modes.c_lflag) & ICANON) { 4804 /* 4805 * Canonical mode is changing state; switch the 4806 * stream head to message-nondiscard or byte-stream 4807 * mode. Also, rerun the service procedure so it can 4808 * change its mind about whether to send data 4809 * upstream or not. 4810 */ 4811 if (tp->t_modes.c_lflag & ICANON) { 4812 DEBUG4(("CHANGING TO CANON MODE\n")); 4813 optbuf.so_flags = SO_READOPT|SO_MREADOFF; 4814 optbuf.so_readopt = RMSGN; 4815 4816 /* 4817 * if there is a pending raw mode timeout, 4818 * clear it 4819 */ 4820 4821 /* 4822 * Clear VMIN/VTIME state, cancel timers 4823 */ 4824 vmin_satisfied(q, tp, 0); 4825 } else { 4826 DEBUG4(("CHANGING TO RAW MODE\n")); 4827 optbuf.so_flags = SO_READOPT|SO_MREADON; 4828 optbuf.so_readopt = RNORM; 4829 } 4830 } 4831 if ((oldmodep->c_lflag ^ tp->t_modes.c_lflag) & TOSTOP) { 4832 /* 4833 * The "stop on background write" bit is changing. 4834 */ 4835 if (tp->t_modes.c_lflag & TOSTOP) 4836 optbuf.so_flags |= SO_TOSTOP; 4837 else 4838 optbuf.so_flags |= SO_TONSTOP; 4839 } 4840 if (optbuf.so_flags != 0) { 4841 if ((bp = allocb(sizeof (struct stroptions), BPRI_HI)) == 4842 NULL) { 4843 return (-1); 4844 } 4845 *(struct stroptions *)bp->b_wptr = optbuf; 4846 bp->b_wptr += sizeof (struct stroptions); 4847 bp->b_datap->db_type = M_SETOPTS; 4848 DEBUG4(("M_SETOPTS to stream head\n")); 4849 putnext(q, bp); 4850 } 4851 return (0); 4852 } 4853 4854 4855 /* 4856 * Called when an M_IOCACK message is seen on the read queue; 4857 * modifies the data being returned, if necessary, and passes the 4858 * reply up. 4859 */ 4860 static void 4861 ldterm_ioctl_reply(queue_t *q, mblk_t *mp) 4862 { 4863 ldtermstd_state_t *tp; 4864 struct iocblk *iocp; 4865 4866 iocp = (struct iocblk *)mp->b_rptr; 4867 tp = (ldtermstd_state_t *)q->q_ptr; 4868 4869 switch (iocp->ioc_cmd) { 4870 4871 case TCGETS: 4872 { 4873 /* 4874 * Get current parameters and return them to 4875 * stream head eventually. 4876 */ 4877 struct termios *cb = 4878 (struct termios *)mp->b_cont->b_rptr; 4879 4880 /* 4881 * cflag has cflags sent upstream by the 4882 * driver 4883 */ 4884 tcflag_t cflag = cb->c_cflag; 4885 4886 *cb = tp->t_amodes; 4887 if (cflag != 0) 4888 cb->c_cflag = cflag; /* set by driver */ 4889 break; 4890 } 4891 4892 case TCGETA: 4893 { 4894 /* 4895 * Old-style "ioctl" to get current 4896 * parameters and return them to stream head 4897 * eventually. 4898 */ 4899 struct termio *cb = 4900 (struct termio *)mp->b_cont->b_rptr; 4901 4902 cb->c_iflag = tp->t_amodes.c_iflag; /* all except the */ 4903 cb->c_oflag = tp->t_amodes.c_oflag; /* cb->c_cflag */ 4904 cb->c_lflag = tp->t_amodes.c_lflag; 4905 4906 if (cb->c_cflag == 0) /* not set by driver */ 4907 cb->c_cflag = tp->t_amodes.c_cflag; 4908 4909 cb->c_line = 0; 4910 bcopy(tp->t_amodes.c_cc, cb->c_cc, NCC); 4911 break; 4912 } 4913 } 4914 putnext(q, mp); 4915 } 4916 4917 4918 /* 4919 * A VMIN/VTIME request has been satisfied. Cancel outstanding timers 4920 * if they exist, clear TS_MREAD state, and send upstream. If a NULL 4921 * queue ptr is passed, just reset VMIN/VTIME state. 4922 */ 4923 static void 4924 vmin_satisfied(queue_t *q, ldtermstd_state_t *tp, int sendup) 4925 { 4926 ASSERT(q); 4927 if (tp->t_vtid != 0) { 4928 DEBUG4(("vmin_satisfied: cancelled timer id %d\n", tp->t_vtid)); 4929 (void) quntimeout(q, tp->t_vtid); 4930 tp->t_vtid = 0; 4931 } 4932 if (sendup) { 4933 if (tp->t_msglen == 0 && V_MIN) { 4934 /* EMPTY */ 4935 DEBUG4(("vmin_satisfied: data swiped, msglen = 0\n")); 4936 } else { 4937 if ((!q->q_first) || 4938 (q->q_first->b_datap->db_type != M_DATA) || 4939 (tp->t_msglen >= LDCHUNK)) { 4940 ldterm_msg_upstream(q, tp); 4941 DEBUG4(("vmin_satisfied: delivering data\n")); 4942 } 4943 } 4944 } else { 4945 /* EMPTY */ 4946 DEBUG4(("vmin_satisfied: VMIN/TIME state reset\n")); 4947 } 4948 tp->t_state &= ~TS_MREAD; 4949 } 4950 4951 static void 4952 vmin_settimer(queue_t *q) 4953 { 4954 ldtermstd_state_t *tp; 4955 4956 tp = (ldtermstd_state_t *)q->q_ptr; 4957 4958 /* 4959 * Don't start any time bombs. 4960 */ 4961 if (tp->t_state & TS_CLOSE) 4962 return; 4963 4964 /* 4965 * tp->t_vtid should NOT be set here unless VMIN > 0 and 4966 * VTIME > 0. 4967 */ 4968 if (tp->t_vtid) { 4969 if (V_MIN && V_TIME) { 4970 /* EMPTY */ 4971 DEBUG4(("vmin_settimer: timer restarted, old tid=%d\n", 4972 tp->t_vtid)); 4973 } else { 4974 /* EMPTY */ 4975 DEBUG4(("vmin_settimer: tid = %d was still active!\n", 4976 tp->t_vtid)); 4977 } 4978 (void) quntimeout(q, tp->t_vtid); 4979 tp->t_vtid = 0; 4980 } 4981 tp->t_vtid = qtimeout(q, vmin_timed_out, q, 4982 (clock_t)(V_TIME * (hz / 10))); 4983 DEBUG4(("vmin_settimer: timer started, tid = %d\n", tp->t_vtid)); 4984 } 4985 4986 4987 /* 4988 * BRRrrringgg!! VTIME was satisfied instead of VMIN 4989 */ 4990 static void 4991 vmin_timed_out(void *arg) 4992 { 4993 queue_t *q = arg; 4994 ldtermstd_state_t *tp; 4995 4996 tp = (ldtermstd_state_t *)q->q_ptr; 4997 4998 DEBUG4(("vmin_timed_out: tid = %d\n", tp->t_vtid)); 4999 /* don't call untimeout now that we are in the timeout */ 5000 tp->t_vtid = 0; 5001 vmin_satisfied(q, tp, 1); 5002 } 5003 5004 5005 /* 5006 * Routine to adjust termios flags to be processed by the line 5007 * discipline. Driver below sends a termios structure, with the flags 5008 * the driver intends to process. XOR'ing the driver sent termios 5009 * structure with current termios structure with the default values 5010 * (or set by ioctls from userland), we come up with a new termios 5011 * structrue, the flags of which will be used by the line discipline 5012 * in processing input and output. On return from this routine, we 5013 * will have the following fields set in tp structure --> 5014 * tp->t_modes: modes the line discipline will process tp->t_amodes: 5015 * modes the user process thinks the line discipline is processing 5016 */ 5017 5018 static void 5019 ldterm_adjust_modes(ldtermstd_state_t *tp) 5020 { 5021 5022 DEBUG6(("original iflag = %o\n", tp->t_modes.c_iflag)); 5023 tp->t_modes.c_iflag = tp->t_amodes.c_iflag & ~(tp->t_dmodes.c_iflag); 5024 tp->t_modes.c_oflag = tp->t_amodes.c_oflag & ~(tp->t_dmodes.c_oflag); 5025 tp->t_modes.c_lflag = tp->t_amodes.c_lflag & ~(tp->t_dmodes.c_lflag); 5026 DEBUG6(("driver iflag = %o\n", tp->t_dmodes.c_iflag)); 5027 DEBUG6(("apparent iflag = %o\n", tp->t_amodes.c_iflag)); 5028 DEBUG6(("effective iflag = %o\n", tp->t_modes.c_iflag)); 5029 5030 /* No negotiation of clfags c_cc array special characters */ 5031 /* 5032 * Copy from amodes to modes already done by TCSETA/TCSETS 5033 * code 5034 */ 5035 } 5036 5037 5038 /* 5039 * Erase one multi-byte character. If TS_MEUC is set AND this 5040 * is a multi-byte character, then this should be called instead of 5041 * ldterm_erase. "ldterm_erase" will handle ASCII nicely, thank you. 5042 * 5043 * We'd better be pointing to the last byte. If we aren't, it will get 5044 * screwed up. 5045 */ 5046 static void 5047 ldterm_csi_erase(queue_t *q, size_t ebsize, ldtermstd_state_t *tp) 5048 { 5049 int i, ung; 5050 uchar_t *p, *bottom; 5051 uchar_t u8[LDTERM_CS_MAX_BYTE_LENGTH]; 5052 int c; 5053 int j; 5054 int len; 5055 5056 if (tp->t_eucleft) { 5057 /* XXX Ick. We're in the middle of an EUC! */ 5058 /* What to do now? */ 5059 ldterm_eucwarn(tp); 5060 return; /* ignore it??? */ 5061 } 5062 bottom = tp->t_eucp_mp->b_rptr; 5063 p = tp->t_eucp - 1; /* previous byte */ 5064 if (p < bottom) 5065 return; 5066 ung = 1; /* number of bytes to un-get from buffer */ 5067 /* 5068 * go through the buffer until we find the beginning of the 5069 * multi-byte char. 5070 */ 5071 while ((*p == 0) && (p > bottom)) { 5072 p--; 5073 ++ung; 5074 } 5075 5076 /* 5077 * Now, "ung" is the number of bytes to unget from the buffer 5078 * and "*p" is the disp width of it. Fool "ldterm_rubout" 5079 * into thinking we're rubbing out ASCII characters. Do that 5080 * for the display width of the character. 5081 * 5082 * Also we accumulate bytes of the character so that if the character 5083 * is a UTF-8 character, we will get the display width of the UTF-8 5084 * character. 5085 */ 5086 if (ung >= LDTERM_CS_MAX_BYTE_LENGTH) { 5087 j = len = LDTERM_CS_MAX_BYTE_LENGTH; 5088 } else { 5089 j = len = ung; 5090 } 5091 for (i = 0; i < ung; i++) { /* remove from buf */ 5092 if ((c = ldterm_unget(tp)) != (-1)) { 5093 ldterm_trim(tp); 5094 if (j > 0) 5095 u8[--j] = (uchar_t)c; 5096 } 5097 } 5098 if (*p == UNKNOWN_WIDTH) { 5099 if (tp->t_csdata.codeset_type == LDTERM_CS_TYPE_UTF8) { 5100 *p = ldterm_utf8_width(u8, len); 5101 } else { 5102 *p = 1; 5103 } 5104 } 5105 for (i = 0; i < (int)*p; i++) /* remove from screen */ 5106 ldterm_rubout(' ', q, ebsize, tp); 5107 /* 5108 * Adjust the parallel array pointer. Zero out the contents 5109 * of parallel array for this position, just to make sure... 5110 */ 5111 tp->t_eucp = p; 5112 *p = 0; 5113 } 5114 5115 5116 /* 5117 * This is kind of a safety valve. Whenever we see a bad sequence 5118 * come up, we call eucwarn. It just tallies the junk until a 5119 * threshold is reached. Then it prints ONE message on the console 5120 * and not any more. Hopefully, we can catch garbage; maybe it will 5121 * be useful to somebody. 5122 */ 5123 static void 5124 ldterm_eucwarn(ldtermstd_state_t *tp) 5125 { 5126 ++tp->t_eucwarn; 5127 #ifdef DEBUG 5128 if ((tp->t_eucwarn > EUC_WARNCNT) && !(tp->t_state & TS_WARNED)) { 5129 cmn_err(CE_WARN, 5130 "ldterm: tty at addr %p in multi-byte mode --", 5131 (void *)tp); 5132 cmn_err(CE_WARN, 5133 "Over %d bad EUC characters this session", EUC_WARNCNT); 5134 tp->t_state |= TS_WARNED; 5135 } 5136 #endif 5137 } 5138 5139 5140 /* 5141 * Copy an "eucioc_t" structure. We use the structure with 5142 * incremented values for Codesets 2 & 3. The specification in 5143 * eucioctl is that the sames values as the CSWIDTH definition at 5144 * user level are passed to us. When we copy it "in" to ourselves, we 5145 * do the increment. That allows us to avoid treating each character 5146 * set separately for "t_eucleft" purposes. When we copy it "out" to 5147 * return it to the user, we decrement the values so the user gets 5148 * what it expects, and it matches CSWIDTH in the environment (if 5149 * things are consistent!). 5150 */ 5151 static void 5152 cp_eucwioc(eucioc_t *from, eucioc_t *to, int dir) 5153 { 5154 bcopy(from, to, EUCSIZE); 5155 if (dir == EUCOUT) { /* copying out to user */ 5156 if (to->eucw[2]) 5157 --to->eucw[2]; 5158 if (to->eucw[3]) 5159 --to->eucw[3]; 5160 } else { /* copying in */ 5161 if (to->eucw[2]) 5162 ++to->eucw[2]; 5163 if (to->eucw[3]) 5164 ++to->eucw[3]; 5165 } 5166 } 5167 5168 5169 /* 5170 * Take the first byte of a multi-byte, or an ASCII char. Return its 5171 * codeset. If it's NOT the first byte of an EUC, then the return 5172 * value may be garbage, as it's probably not SS2 or SS3, and 5173 * therefore must be in codeset 1. Another bizarre catch here is the 5174 * fact that we don't do anything about the "C1" control codes. In 5175 * real life, we should; but nobody's come up with a good way of 5176 * treating them. 5177 */ 5178 5179 static int 5180 ldterm_codeset(uchar_t codeset_type, uchar_t c) 5181 { 5182 5183 if (ISASCII(c)) 5184 return (0); 5185 5186 if (codeset_type != LDTERM_CS_TYPE_EUC) 5187 return (1); 5188 5189 switch (c) { 5190 case SS2: 5191 return (2); 5192 case SS3: 5193 return (3); 5194 default: 5195 return (1); 5196 } 5197 } 5198 5199 /* The following two functions are additional EUC codeset specific methods. */ 5200 /* 5201 * ldterm_dispwidth - Take the first byte of an EUC (or ASCII) and 5202 * return the display width. Since this is intended mostly for 5203 * multi-byte handling, it returns EUC_TWIDTH for tabs so they can be 5204 * differentiated from EUC characters (assumption: EUC require fewer 5205 * than 255 columns). Also, if it's a backspace and !flag, it 5206 * returns EUC_BSWIDTH. Newline & CR also depend on flag. This 5207 * routine SHOULD be cleaner than this, but we have the situation 5208 * where we may or may not be counting control characters as having a 5209 * column width. Therefore, the computation of ASCII is pretty messy. 5210 * The caller will be storing the value, and then switching on it 5211 * when it's used. We really should define the EUC_TWIDTH and other 5212 * constants in a header so that the routine could be used in other 5213 * modules in the kernel. 5214 */ 5215 static int 5216 __ldterm_dispwidth_euc(uchar_t c, void *p, int mode) 5217 { 5218 ldtermstd_state_t *tp = (ldtermstd_state_t *)p; 5219 5220 if (ISASCII(c)) { 5221 if (c <= '\037') { 5222 switch (c) { 5223 case '\t': 5224 return (EUC_TWIDTH); 5225 case '\b': 5226 return (mode ? 2 : EUC_BSWIDTH); 5227 case '\n': 5228 return (EUC_NLWIDTH); 5229 case '\r': 5230 return (mode ? 2 : EUC_CRWIDTH); 5231 default: 5232 return (mode ? 2 : 0); 5233 } 5234 } 5235 return (1); 5236 } 5237 switch (c) { 5238 case SS2: 5239 return (tp->eucwioc.scrw[2]); 5240 case SS3: 5241 return (tp->eucwioc.scrw[3]); 5242 default: 5243 return (tp->eucwioc.scrw[1]); 5244 } 5245 } 5246 5247 /* 5248 * ldterm_memwidth_euc - Take the first byte of an EUC (or an ASCII char) 5249 * and return its memory width. The routine could have been 5250 * implemented to use only the codeset number, but that would require 5251 * the caller to have that value available. Perhaps the user doesn't 5252 * want to make the extra call or keep the value of codeset around. 5253 * Therefore, we use the actual character with which they're 5254 * concerned. This should never be called with anything but the 5255 * first byte of an EUC, otherwise it will return a garbage value. 5256 */ 5257 static int 5258 __ldterm_memwidth_euc(uchar_t c, void *p) 5259 { 5260 ldtermstd_state_t *tp = (ldtermstd_state_t *)p; 5261 5262 if (ISASCII(c)) 5263 return (1); 5264 switch (c) { 5265 case SS2: 5266 return (tp->eucwioc.eucw[2]); 5267 case SS3: 5268 return (tp->eucwioc.eucw[3]); 5269 default: 5270 return (tp->eucwioc.eucw[1]); 5271 } 5272 } 5273 5274 5275 /* The following two functions are PCCS codeset specific methods. */ 5276 static int 5277 __ldterm_dispwidth_pccs(uchar_t c, void *p, int mode) 5278 { 5279 ldtermstd_state_t *tp = (ldtermstd_state_t *)p; 5280 int i; 5281 5282 if (ISASCII(c)) { 5283 if (c <= '\037') { 5284 switch (c) { 5285 case '\t': 5286 return (EUC_TWIDTH); 5287 case '\b': 5288 return (mode ? 2 : EUC_BSWIDTH); 5289 case '\n': 5290 return (EUC_NLWIDTH); 5291 case '\r': 5292 return (mode ? 2 : EUC_CRWIDTH); 5293 default: 5294 return (mode ? 2 : 0); 5295 } 5296 } 5297 return (1); 5298 } 5299 5300 for (i = 0; i < tp->t_csdata.csinfo_num; i++) { 5301 if (c >= tp->t_csdata.eucpc_data[i].msb_start && 5302 c <= tp->t_csdata.eucpc_data[i].msb_end) 5303 return (tp->t_csdata.eucpc_data[i].screen_width); 5304 } 5305 5306 /* 5307 * If this leading byte is not in the range list, either provided 5308 * locale data is not sufficient or we encountered an invalid 5309 * character. We return 1 in this case as a fallback value. 5310 */ 5311 return (1); 5312 } 5313 5314 static int 5315 __ldterm_memwidth_pccs(uchar_t c, void *p) 5316 { 5317 ldtermstd_state_t *tp = (ldtermstd_state_t *)p; 5318 int i; 5319 5320 for (i = 0; i < tp->t_csdata.csinfo_num; i++) { 5321 if (c >= tp->t_csdata.eucpc_data[i].msb_start && 5322 c <= tp->t_csdata.eucpc_data[i].msb_end) 5323 return (tp->t_csdata.eucpc_data[i].byte_length); 5324 } 5325 5326 /* 5327 * If this leading byte is not in the range list, either provided 5328 * locale data is not sufficient or we encountered an invalid 5329 * character. We return 1 in this case as a fallback value. 5330 */ 5331 return (1); 5332 } 5333 5334 5335 /* The following two functions are UTF-8 codeset specific methods. */ 5336 static int 5337 __ldterm_dispwidth_utf8(uchar_t c, void *p, int mode) 5338 { 5339 ldtermstd_state_t *tp = (ldtermstd_state_t *)p; 5340 5341 if (ISASCII(c)) { 5342 if (c <= '\037') { 5343 switch (c) { 5344 case '\t': 5345 return (EUC_TWIDTH); 5346 case '\b': 5347 return (mode ? 2 : EUC_BSWIDTH); 5348 case '\n': 5349 return (EUC_NLWIDTH); 5350 case '\r': 5351 return (mode ? 2 : EUC_CRWIDTH); 5352 default: 5353 return (mode ? 2 : 0); 5354 } 5355 } 5356 return (1); 5357 } 5358 5359 /* This is to silence the lint. */ 5360 if (tp->t_csdata.codeset_type != LDTERM_CS_TYPE_UTF8) 5361 return (1); 5362 5363 /* 5364 * If it is a valid leading byte of a UTF-8 character, we set 5365 * the width as 'UNKNOWN_WIDTH' for now. We need to have all 5366 * the bytes to figure out the display width. 5367 */ 5368 if (c >= (uchar_t)0xc0 && c <= (uchar_t)0xfd) 5369 return (UNKNOWN_WIDTH); 5370 5371 /* 5372 * If it is an invalid leading byte, we just do our best by 5373 * giving the display width of 1. 5374 */ 5375 return (1); 5376 } 5377 5378 5379 static int 5380 __ldterm_memwidth_utf8(uchar_t c, void *p) 5381 { 5382 ldtermstd_state_t *tp = (ldtermstd_state_t *)p; 5383 int len; 5384 5385 /* 5386 * If the codeset type doesn't match, we treat them as 5387 * an illegal character and return 1. 5388 */ 5389 if (tp->t_csdata.codeset_type != LDTERM_CS_TYPE_UTF8) 5390 return (1); 5391 5392 len = u8_number_of_bytes[c]; 5393 5394 /* 5395 * If this is a start of an illegal character, we treat 5396 * such as an 1 byte character and screen out. 5397 */ 5398 return ((len <= 0) ? 1 : len); 5399 } 5400 5401 static uchar_t 5402 ldterm_utf8_width(uchar_t *u8, int length) 5403 { 5404 int i; 5405 int j; 5406 uint_t intcode = 0; 5407 5408 if (length == 0) 5409 return ('\0'); 5410 5411 j = u8_number_of_bytes[u8[0]] - 1; 5412 5413 /* 5414 * If the UTF-8 character is out of UTF-16 code range, or, 5415 * if it is either an ASCII character or an invalid leading byte for 5416 * a UTF-8 character, return 1. 5417 */ 5418 if (length > 4 || j <= 0) 5419 return ('\1'); 5420 5421 intcode = u8[0] & u8_masks_tbl[j]; 5422 for (i = 1; j > 0; j--, i++) { 5423 /* 5424 * The following additional checking is needed to conform to 5425 * the "UTF-8 Corrigendum" introduced at the Unicode 3.1 and 5426 * then updated one more time at the Unicode 3.2. 5427 */ 5428 if (i == 1) { 5429 if (u8[i] < u8_valid_min_2nd_byte[u8[0]] || 5430 u8[i] > u8_valid_max_2nd_byte[u8[0]]) 5431 return ('\1'); 5432 } else if (u8[i] < (uchar_t)LDTERM_CS_TYPE_UTF8_MIN_BYTE || 5433 u8[i] > (uchar_t)LDTERM_CS_TYPE_UTF8_MAX_BYTE) 5434 return ('\1'); 5435 5436 /* 5437 * All subsequent bytes of UTF-8 character has the following 5438 * binary encoding: 5439 * 5440 * 10xx xxxx 5441 * 5442 * hence left shift six bits to make space and then get 5443 * six bits from the new byte. 5444 */ 5445 intcode = (intcode << LDTERM_CS_TYPE_UTF8_SHIFT_BITS) | 5446 (u8[i] & LDTERM_CS_TYPE_UTF8_BIT_MASK); 5447 } 5448 5449 i = 0; 5450 if (intcode <= LDTERM_CS_TYPE_UTF8_MAX_P00) { 5451 /* Basic Multilingual Plane. */ 5452 i = intcode / 4; 5453 j = intcode % 4; 5454 switch (j) { 5455 case 0: 5456 i = ldterm_ucode[0][i].u0; 5457 break; 5458 case 1: 5459 i = ldterm_ucode[0][i].u1; 5460 break; 5461 case 2: 5462 i = ldterm_ucode[0][i].u2; 5463 break; 5464 case 3: 5465 i = ldterm_ucode[0][i].u3; 5466 break; 5467 } 5468 } else if (intcode <= LDTERM_CS_TYPE_UTF8_MAX_P01) { 5469 /* Secondary Multilingual Plane. */ 5470 intcode = intcode & (uint_t)0xffff; 5471 i = intcode / 4; 5472 j = intcode % 4; 5473 switch (j) { 5474 case 0: 5475 i = ldterm_ucode[1][i].u0; 5476 break; 5477 case 1: 5478 i = ldterm_ucode[1][i].u1; 5479 break; 5480 case 2: 5481 i = ldterm_ucode[1][i].u2; 5482 break; 5483 case 3: 5484 i = ldterm_ucode[1][i].u3; 5485 break; 5486 } 5487 } else if ((intcode >= LDTERM_CS_TYPE_UTF8_MIN_CJKEXTB && 5488 intcode <= LDTERM_CS_TYPE_UTF8_MAX_CJKEXTB) || 5489 (intcode >= LDTERM_CS_TYPE_UTF8_MIN_CJKCOMP && 5490 intcode <= LDTERM_CS_TYPE_UTF8_MAX_CJKCOMP) || 5491 (intcode >= LDTERM_CS_TYPE_UTF8_MIN_P15 && 5492 intcode <= LDTERM_CS_TYPE_UTF8_MAX_P15) || 5493 (intcode >= LDTERM_CS_TYPE_UTF8_MIN_P16 && 5494 intcode <= LDTERM_CS_TYPE_UTF8_MAX_P16)) { 5495 /* 5496 * Supplementary Plane for CJK Ideographs and 5497 * Private Use Planes. 5498 */ 5499 return ('\2'); 5500 } else if ((intcode >= LDTERM_CS_TYPE_UTF8_MIN_P14 && 5501 intcode <= LDTERM_CS_TYPE_UTF8_MAX_P14) || 5502 (intcode >= LDTERM_CS_TYPE_UTF8_MIN_VARSEL && 5503 intcode <= LDTERM_CS_TYPE_UTF8_MAX_VARSEL)) { 5504 /* 5505 * Some Special Purpose Plane characters: 5506 * These are like control characters and not printable. 5507 */ 5508 return ('\0'); 5509 } 5510 5511 /* 5512 * We return the display width of 1 for all character code points 5513 * that we didn't catch from the above logic and also for combining 5514 * and conjoining characters with width value of zero. 5515 * 5516 * In particular, the reason why we are returning 1 for combining 5517 * and conjoining characters is because the GUI-based terminal 5518 * emulators are not yet capable of properly handling such characters 5519 * and in most of the cases, they just treat such characters as if 5520 * they occupy a display cell. If the terminal emulators are capable of 5521 * handling the characters correctly, then, this logic of returning 5522 * 1 should be revisited and changed. See CR 6660526 for more 5523 * details on this. 5524 */ 5525 return ((i == 0) ? '\1' : (uchar_t)i); 5526 } --- EOF ---