term_private.h revision 1.20 1 1.20 js /* $NetBSD: term_private.h,v 1.20 2025/09/23 19:50:43 js Exp $ */
2 1.1 roy
3 1.1 roy /*
4 1.12 roy * Copyright (c) 2009, 2010, 2013, 2020 The NetBSD Foundation, Inc.
5 1.1 roy *
6 1.1 roy * This code is derived from software contributed to The NetBSD Foundation
7 1.1 roy * by Roy Marples.
8 1.1 roy *
9 1.1 roy * Redistribution and use in source and binary forms, with or without
10 1.1 roy * modification, are permitted provided that the following conditions
11 1.1 roy * are met:
12 1.1 roy * 1. Redistributions of source code must retain the above copyright
13 1.1 roy * notice, this list of conditions and the following disclaimer.
14 1.1 roy * 2. Redistributions in binary form must reproduce the above copyright
15 1.1 roy * notice, this list of conditions and the following disclaimer in the
16 1.1 roy * documentation and/or other materials provided with the distribution.
17 1.1 roy *
18 1.1 roy * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
19 1.1 roy * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
20 1.1 roy * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
21 1.1 roy * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
22 1.1 roy * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
23 1.1 roy * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24 1.1 roy * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25 1.1 roy * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26 1.1 roy * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
27 1.1 roy * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 1.1 roy */
29 1.1 roy
30 1.1 roy #ifndef _TERM_PRIVATE_H_
31 1.1 roy #define _TERM_PRIVATE_H_
32 1.1 roy
33 1.1 roy /* This header should only be used by libterminfo, tic and infocmp. */
34 1.1 roy
35 1.1 roy /* The terminfo database structure is private to us,
36 1.1 roy * so it's documented here.
37 1.12 roy *
38 1.12 roy * Version 1 - types 1 and 2.
39 1.1 roy * terminfo defines the largest number as 32767 and the largest
40 1.12 roy * compiled entry as 4093 bytes long. Negative numbers are not allowed.
41 1.1 roy * Thus, we store all numbers as uint16_t, including string length.
42 1.12 roy * We reserve negative numbers -1 and -2 to mean absent or cancelled.
43 1.1 roy * All strings are prefixed by length, including the null terminator.
44 1.1 roy * The largest string length we can handle is 65535 bytes,
45 1.1 roy * including the null terminator.
46 1.2 snj * The largest capability block we can handle is 65535 bytes.
47 1.10 joerg *
48 1.12 roy * Version 2 - type 3
49 1.12 roy * Extends terminfo numbers upto 2147483647 by storing the value as a uint32_t.
50 1.12 roy * This means that we exceed the current terminfo defined limits in every way.
51 1.12 roy *
52 1.12 roy * Type 1 capabilities are defined as:
53 1.10 joerg * header byte (always 1)
54 1.10 joerg * name
55 1.1 roy * description,
56 1.1 roy * cap length, num flags, index, char,
57 1.1 roy * cap length, num numbers, index, number,
58 1.1 roy * cap length, num strings, index, string,
59 1.10 joerg * cap length, num undefined caps, name, type (char), flag, number, string
60 1.10 joerg *
61 1.12 roy * Type 2 entries are aliases and defined as:
62 1.10 joerg * header byte (always 2)
63 1.10 joerg * 32bit id of the corresponding terminal in the file
64 1.10 joerg * name
65 1.10 joerg *
66 1.12 roy * Type 3 extends Type 1 so that it can store terminfo numbers
67 1.12 roy * as uint32_t. All other numerics are still stored as uint16_t.
68 1.12 roy *
69 1.10 joerg * The database itself is created using cdbw(3) and the numbers are
70 1.1 roy * always stored as little endian.
71 1.1 roy */
72 1.1 roy
73 1.20 js #include <sys/endian.h>
74 1.1 roy #include <sys/types.h>
75 1.14 christos #include <assert.h>
76 1.14 christos #include <limits.h>
77 1.1 roy
78 1.1 roy #define _TERMINFO
79 1.12 roy #define TERMINFO_RTYPE_O1 1
80 1.12 roy #define TERMINFO_ALIAS 2
81 1.12 roy #define TERMINFO_RTYPE 3
82 1.1 roy
83 1.17 roy /* , and | are the two print characters now allowed
84 1.17 roy * in terminfo aliases or long descriptions.
85 1.17 roy * As | is generally used to delimit aliases inside the
86 1.17 roy * description, we use a comma. */
87 1.17 roy #define TERMINFO_VDELIM ','
88 1.17 roy #define TERMINFO_VDELIMSTR ","
89 1.17 roy
90 1.1 roy /* We use the same ncurses tic macros so that our data is identical
91 1.1 roy * when a caller uses the long name macros to access te terminfo data
92 1.1 roy * directly. */
93 1.1 roy #define ABSENT_BOOLEAN ((signed char)-1) /* 255 */
94 1.1 roy #define ABSENT_NUMERIC (-1)
95 1.1 roy #define ABSENT_STRING (char *)0
96 1.1 roy #define CANCELLED_BOOLEAN ((signed char)-2) /* 254 */
97 1.1 roy #define CANCELLED_NUMERIC (-2)
98 1.1 roy #define CANCELLED_STRING (char *)(-1)
99 1.1 roy #define VALID_BOOLEAN(s) ((unsigned char)(s) <= 1) /* reject "-1" */
100 1.1 roy #define VALID_NUMERIC(s) ((s) >= 0)
101 1.1 roy #define VALID_STRING(s) ((s) != CANCELLED_STRING && (s) != ABSENT_STRING)
102 1.1 roy
103 1.8 roy typedef struct {
104 1.1 roy const char *id;
105 1.1 roy char type;
106 1.1 roy char flag;
107 1.12 roy int num;
108 1.1 roy const char *str;
109 1.1 roy } TERMUSERDEF;
110 1.1 roy
111 1.8 roy typedef struct {
112 1.1 roy int fildes;
113 1.1 roy /* We need to expose these so that the macros work */
114 1.5 roy const char *name;
115 1.5 roy const char *desc;
116 1.4 he signed char *flags;
117 1.12 roy int *nums;
118 1.1 roy const char **strs;
119 1.1 roy /* Storage area for terminfo data */
120 1.1 roy char *_area;
121 1.5 roy size_t _arealen;
122 1.1 roy size_t _nuserdefs;
123 1.1 roy TERMUSERDEF *_userdefs;
124 1.1 roy /* So we don't rely on the global ospeed */
125 1.1 roy short _ospeed;
126 1.1 roy /* Output buffer for tparm */
127 1.1 roy char *_buf;
128 1.1 roy size_t _buflen;
129 1.1 roy size_t _bufpos;
130 1.1 roy /* A-Z static variables for tparm */
131 1.1 roy long _snums[26];
132 1.3 roy /* aliases of the terminal, | separated */
133 1.5 roy const char *_alias;
134 1.1 roy } TERMINAL;
135 1.1 roy
136 1.1 roy extern const char * _ti_database;
137 1.1 roy
138 1.1 roy ssize_t _ti_flagindex(const char *);
139 1.1 roy ssize_t _ti_numindex(const char *);
140 1.1 roy ssize_t _ti_strindex(const char *);
141 1.1 roy const char * _ti_flagid(ssize_t);
142 1.1 roy const char * _ti_numid(ssize_t);
143 1.1 roy const char * _ti_strid(ssize_t);
144 1.1 roy int _ti_getterm(TERMINAL *, const char *, int);
145 1.1 roy void _ti_setospeed(TERMINAL *);
146 1.1 roy
147 1.6 roy /* libterminfo can compile terminfo strings too */
148 1.6 roy #define TIC_WARNING (1 << 0)
149 1.6 roy #define TIC_DESCRIPTION (1 << 1)
150 1.6 roy #define TIC_ALIAS (1 << 2)
151 1.6 roy #define TIC_COMMENT (1 << 3)
152 1.6 roy #define TIC_EXTRA (1 << 4)
153 1.13 christos #define TIC_COMPAT_V1 (1 << 5)
154 1.6 roy
155 1.8 roy typedef struct {
156 1.6 roy char *buf;
157 1.6 roy size_t buflen;
158 1.6 roy size_t bufpos;
159 1.6 roy size_t entries;
160 1.6 roy } TBUF;
161 1.6 roy
162 1.8 roy typedef struct {
163 1.6 roy char *name;
164 1.6 roy char *alias;
165 1.6 roy char *desc;
166 1.13 christos int rtype;
167 1.6 roy TBUF flags;
168 1.6 roy TBUF nums;
169 1.6 roy TBUF strs;
170 1.6 roy TBUF extras;
171 1.6 roy } TIC;
172 1.6 roy
173 1.13 christos #define _ti_numsize(tic) \
174 1.13 christos ((tic)->rtype == TERMINFO_RTYPE_O1 ? sizeof(uint16_t) : sizeof(uint32_t))
175 1.13 christos
176 1.18 roy int _ti_promote(TIC *);
177 1.6 roy char *_ti_grow_tbuf(TBUF *, size_t);
178 1.7 roy char *_ti_get_token(char **, char);
179 1.14 christos const char *_ti_find_cap(TIC *, TBUF *, char, short);
180 1.14 christos const char *_ti_find_extra(TIC *, TBUF *, const char *);
181 1.13 christos char *_ti_getname(int, const char *);
182 1.13 christos size_t _ti_store_extra(TIC *, int, const char *, char, char, int,
183 1.13 christos const char *, size_t, int);
184 1.6 roy TIC *_ti_compile(char *, int);
185 1.6 roy ssize_t _ti_flatten(uint8_t **, const TIC *);
186 1.6 roy void _ti_freetic(TIC *);
187 1.11 roy
188 1.14 christos int _ti_encode_buf_id_num(TBUF *, int, int, size_t);
189 1.14 christos int _ti_encode_buf_id_count_str(TBUF *, int, const void *, size_t);
190 1.14 christos int _ti_encode_buf_id_flags(TBUF *, int, int);
191 1.14 christos
192 1.11 roy #define TPARM_MAX 9 /* not likely to change */
193 1.11 roy int _ti_parm_analyse(const char *, int *, int);
194 1.14 christos
195 1.14 christos static __inline int
196 1.14 christos _ti_decode_16(const char **cap)
197 1.14 christos {
198 1.16 roy int num = (int16_t)le16dec(*cap);
199 1.14 christos
200 1.14 christos *cap += sizeof(uint16_t);
201 1.14 christos return num;
202 1.14 christos }
203 1.15 roy
204 1.14 christos static __inline int
205 1.14 christos _ti_decode_32(const char **cap)
206 1.14 christos {
207 1.16 roy int num = (int32_t)le32dec(*cap);
208 1.14 christos
209 1.14 christos *cap += sizeof(uint32_t);
210 1.14 christos return num;
211 1.14 christos }
212 1.14 christos
213 1.14 christos static __inline int
214 1.14 christos _ti_decode_num(const char **cap, int rtype)
215 1.14 christos {
216 1.14 christos if (rtype == TERMINFO_RTYPE_O1) {
217 1.14 christos return _ti_decode_16(cap);
218 1.14 christos } else {
219 1.14 christos return _ti_decode_32(cap);
220 1.14 christos }
221 1.14 christos }
222 1.14 christos
223 1.14 christos static __inline void
224 1.14 christos _ti_encode_16(char **cap, size_t num)
225 1.14 christos {
226 1.16 roy _DIAGASSERT(num <= UINT16_MAX);
227 1.14 christos le16enc(*cap, (uint16_t)num);
228 1.14 christos *cap += sizeof(uint16_t);
229 1.14 christos }
230 1.15 roy
231 1.14 christos static __inline void
232 1.14 christos _ti_encode_32(char **cap, size_t num)
233 1.14 christos {
234 1.16 roy _DIAGASSERT(num <= UINT32_MAX);
235 1.14 christos le32enc(*cap, (uint32_t)num);
236 1.14 christos *cap += sizeof(uint32_t);
237 1.14 christos }
238 1.14 christos
239 1.14 christos static __inline void
240 1.14 christos _ti_encode_str(char **cap, const void *buf, size_t len)
241 1.14 christos {
242 1.14 christos memcpy(*cap, buf, len);
243 1.14 christos *cap += len;
244 1.14 christos }
245 1.14 christos
246 1.14 christos static __inline void
247 1.14 christos _ti_encode_count_str(char **cap, const char *name, size_t len)
248 1.14 christos {
249 1.14 christos _ti_encode_16(cap, (uint16_t)len);
250 1.14 christos if (name == NULL)
251 1.14 christos return;
252 1.14 christos _ti_encode_str(cap, name, len);
253 1.14 christos }
254 1.14 christos
255 1.14 christos static __inline void
256 1.14 christos _ti_encode_buf_16(TBUF *tbuf, size_t num)
257 1.14 christos {
258 1.16 roy _DIAGASSERT(num <= UINT16_MAX);
259 1.14 christos le16enc(tbuf->buf + tbuf->bufpos, (uint16_t)num);
260 1.14 christos tbuf->bufpos += sizeof(uint16_t);
261 1.14 christos }
262 1.14 christos
263 1.14 christos static __inline void
264 1.14 christos _ti_encode_buf_32(TBUF *tbuf, size_t num)
265 1.14 christos {
266 1.16 roy _DIAGASSERT(num <= UINT32_MAX);
267 1.14 christos le32enc(tbuf->buf + tbuf->bufpos, (uint32_t)num);
268 1.14 christos tbuf->bufpos += sizeof(uint32_t);
269 1.14 christos }
270 1.14 christos
271 1.14 christos static __inline void
272 1.14 christos _ti_encode_buf_count_str(TBUF *tbuf, const void *buf, size_t len)
273 1.14 christos {
274 1.14 christos _ti_encode_buf_16(tbuf, len);
275 1.14 christos memcpy(tbuf->buf + tbuf->bufpos, buf, len);
276 1.14 christos tbuf->bufpos += len;
277 1.14 christos }
278 1.14 christos
279 1.14 christos static __inline void
280 1.19 roy _ti_encode_buf_num(TBUF *tbuf, int num, int rtype)
281 1.14 christos {
282 1.14 christos if (rtype == TERMINFO_RTYPE_O1) {
283 1.16 roy if (num > INT16_MAX)
284 1.16 roy num = INT16_MAX;
285 1.19 roy _ti_encode_buf_16(tbuf, (uint16_t)num);
286 1.14 christos } else {
287 1.19 roy if (num > INT32_MAX)
288 1.19 roy num = INT32_MAX;
289 1.19 roy _ti_encode_buf_32(tbuf, (uint32_t)num);
290 1.14 christos }
291 1.14 christos }
292 1.14 christos
293 1.1 roy #endif
294