master
1/*-
2 * SPDX-License-Identifier: BSD-2-Clause
3 *
4 * Copyright (c) 2009 The FreeBSD Foundation
5 *
6 * This software was developed by Ed Schouten under sponsorship from the
7 * FreeBSD Foundation.
8 *
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted provided that the following conditions
11 * are met:
12 * 1. Redistributions of source code must retain the above copyright
13 * notice, this list of conditions and the following disclaimer.
14 * 2. Redistributions in binary form must reproduce the above copyright
15 * notice, this list of conditions and the following disclaimer in the
16 * documentation and/or other materials provided with the distribution.
17 *
18 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
19 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
22 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28 * SUCH DAMAGE.
29 */
30
31#ifndef _SYS_TERMINAL_H_
32#define _SYS_TERMINAL_H_
33
34#include <sys/param.h>
35#include <sys/_lock.h>
36#include <sys/_mutex.h>
37#include <sys/cons.h>
38#include <sys/linker_set.h>
39#include <sys/ttycom.h>
40
41#include <teken/teken.h>
42
43#include "opt_syscons.h"
44#include "opt_teken.h"
45
46struct terminal;
47struct thread;
48struct tty;
49
50/*
51 * The terminal layer is an abstraction on top of the TTY layer and the
52 * console interface. It can be used by system console drivers to
53 * easily interact with the kernel console and TTYs.
54 *
55 * Terminals contain terminal emulators, which means console drivers
56 * don't need to implement their own terminal emulator. The terminal
57 * emulator deals with UTF-8 exclusively. This means that term_char_t,
58 * the data type used to store input/output characters will always
59 * contain Unicode codepoints.
60 *
61 * To save memory usage, the top bits of term_char_t will contain other
62 * attributes, like colors. Right now term_char_t is composed as
63 * follows:
64 *
65 * Bits Meaning
66 * 0-20: Character value
67 * 21-25: Bold, underline, blink, reverse, right part of CJK fullwidth character
68 * 26-28: Foreground color
69 * 29-31: Background color
70 */
71
72typedef uint32_t term_char_t;
73#define TCHAR_CHARACTER(c) ((c) & 0x1fffff)
74#define TCHAR_FORMAT(c) (((c) >> 21) & 0x1f)
75#define TCHAR_FGCOLOR(c) (((c) >> 26) & 0x7)
76#define TCHAR_BGCOLOR(c) (((c) >> 29) & 0x7)
77
78typedef teken_attr_t term_attr_t;
79
80typedef teken_color_t term_color_t;
81#define TCOLOR_FG(c) (((c) & 0x7) << 26)
82#define TCOLOR_BG(c) (((c) & 0x7) << 29)
83#define TCOLOR_LIGHT(c) ((c) | 0x8)
84#define TCOLOR_DARK(c) ((c) & ~0x8)
85
86#define TFORMAT(c) (((c) & 0x1f) << 21)
87
88/* syscons(4) compatible color attributes for foreground text */
89#define FG_BLACK TCOLOR_FG(TC_BLACK)
90#define FG_BLUE TCOLOR_FG(TC_BLUE)
91#define FG_GREEN TCOLOR_FG(TC_GREEN)
92#define FG_CYAN TCOLOR_FG(TC_CYAN)
93#define FG_RED TCOLOR_FG(TC_RED)
94#define FG_MAGENTA TCOLOR_FG(TC_MAGENTA)
95#define FG_BROWN TCOLOR_FG(TC_YELLOW)
96#define FG_LIGHTGREY TCOLOR_FG(TC_WHITE)
97#define FG_DARKGREY (TFORMAT(TF_BOLD) | TCOLOR_FG(TC_BLACK))
98#define FG_LIGHTBLUE (TFORMAT(TF_BOLD) | TCOLOR_FG(TC_BLUE))
99#define FG_LIGHTGREEN (TFORMAT(TF_BOLD) | TCOLOR_FG(TC_GREEN))
100#define FG_LIGHTCYAN (TFORMAT(TF_BOLD) | TCOLOR_FG(TC_CYAN))
101#define FG_LIGHTRED (TFORMAT(TF_BOLD) | TCOLOR_FG(TC_RED))
102#define FG_LIGHTMAGENTA (TFORMAT(TF_BOLD) | TCOLOR_FG(TC_MAGENTA))
103#define FG_YELLOW (TFORMAT(TF_BOLD) | TCOLOR_FG(TC_YELLOW))
104#define FG_WHITE (TFORMAT(TF_BOLD) | TCOLOR_FG(TC_WHITE))
105#define FG_BLINK TFORMAT(TF_BLINK)
106
107/* syscons(4) compatible color attributes for text background */
108#define BG_BLACK TCOLOR_BG(TC_BLACK)
109#define BG_BLUE TCOLOR_BG(TC_BLUE)
110#define BG_GREEN TCOLOR_BG(TC_GREEN)
111#define BG_CYAN TCOLOR_BG(TC_CYAN)
112#define BG_RED TCOLOR_BG(TC_RED)
113#define BG_MAGENTA TCOLOR_BG(TC_MAGENTA)
114#define BG_BROWN TCOLOR_BG(TC_YELLOW)
115#define BG_LIGHTGREY TCOLOR_BG(TC_WHITE)
116#define BG_DARKGREY (TFORMAT(TF_BOLD) | TCOLOR_BG(TC_BLACK))
117#define BG_LIGHTBLUE (TFORMAT(TF_BOLD) | TCOLOR_BG(TC_BLUE))
118#define BG_LIGHTGREEN (TFORMAT(TF_BOLD) | TCOLOR_BG(TC_GREEN))
119#define BG_LIGHTCYAN (TFORMAT(TF_BOLD) | TCOLOR_BG(TC_CYAN))
120#define BG_LIGHTRED (TFORMAT(TF_BOLD) | TCOLOR_BG(TC_RED))
121#define BG_LIGHTMAGENTA (TFORMAT(TF_BOLD) | TCOLOR_BG(TC_MAGENTA))
122#define BG_YELLOW (TFORMAT(TF_BOLD) | TCOLOR_BG(TC_YELLOW))
123#define BG_WHITE (TFORMAT(TF_BOLD) | TCOLOR_BG(TC_WHITE))
124
125#ifndef TERMINAL_NORM_ATTR
126#ifdef SC_NORM_ATTR
127#define TERMINAL_NORM_ATTR SC_NORM_ATTR
128#else
129#define TERMINAL_NORM_ATTR (FG_LIGHTGREY | BG_BLACK)
130#endif
131#endif
132
133#ifndef TERMINAL_KERN_ATTR
134#ifdef SC_KERNEL_CONS_ATTR
135#define TERMINAL_KERN_ATTR SC_KERNEL_CONS_ATTR
136#else
137#define TERMINAL_KERN_ATTR (FG_WHITE | BG_BLACK)
138#endif
139#endif
140
141typedef teken_pos_t term_pos_t;
142typedef teken_rect_t term_rect_t;
143
144typedef void tc_cursor_t(struct terminal *tm, const term_pos_t *p);
145typedef void tc_putchar_t(struct terminal *tm, const term_pos_t *p,
146 term_char_t c);
147typedef void tc_fill_t(struct terminal *tm, const term_rect_t *r,
148 term_char_t c);
149typedef void tc_copy_t(struct terminal *tm, const term_rect_t *r,
150 const term_pos_t *p);
151typedef void tc_pre_input_t(struct terminal *tm);
152typedef void tc_post_input_t(struct terminal *tm);
153typedef void tc_param_t(struct terminal *tm, int cmd, unsigned int arg);
154typedef void tc_done_t(struct terminal *tm);
155
156typedef void tc_cnprobe_t(struct terminal *tm, struct consdev *cd);
157typedef int tc_cngetc_t(struct terminal *tm);
158
159typedef void tc_cngrab_t(struct terminal *tm);
160typedef void tc_cnungrab_t(struct terminal *tm);
161
162typedef void tc_opened_t(struct terminal *tm, int opened);
163typedef int tc_ioctl_t(struct terminal *tm, u_long cmd, caddr_t data,
164 struct thread *td);
165typedef int tc_mmap_t(struct terminal *tm, vm_ooffset_t offset,
166 vm_paddr_t * paddr, int nprot, vm_memattr_t *memattr);
167typedef void tc_bell_t(struct terminal *tm);
168
169struct terminal_class {
170 /* Terminal emulator. */
171 tc_cursor_t *tc_cursor;
172 tc_putchar_t *tc_putchar;
173 tc_fill_t *tc_fill;
174 tc_copy_t *tc_copy;
175 tc_pre_input_t *tc_pre_input;
176 tc_post_input_t *tc_post_input;
177 tc_param_t *tc_param;
178 tc_done_t *tc_done;
179
180 /* Low-level console interface. */
181 tc_cnprobe_t *tc_cnprobe;
182 tc_cngetc_t *tc_cngetc;
183
184 /* DDB & panic handling. */
185 tc_cngrab_t *tc_cngrab;
186 tc_cnungrab_t *tc_cnungrab;
187
188 /* Misc. */
189 tc_opened_t *tc_opened;
190 tc_ioctl_t *tc_ioctl;
191 tc_mmap_t *tc_mmap;
192 tc_bell_t *tc_bell;
193};
194
195struct terminal {
196 const struct terminal_class *tm_class;
197 void *tm_softc;
198 struct mtx tm_mtx;
199 struct tty *tm_tty;
200 teken_t tm_emulator;
201 struct winsize tm_winsize;
202 unsigned int tm_flags;
203#define TF_MUTE 0x1 /* Drop incoming data. */
204#define TF_BELL 0x2 /* Bell needs to be sent. */
205#define TF_CONS 0x4 /* Console device (needs spinlock). */
206 struct consdev *consdev;
207};
208
209#ifdef _KERNEL
210
211struct terminal *terminal_alloc(const struct terminal_class *tc, void *softc);
212void terminal_maketty(struct terminal *tm, const char *fmt, ...);
213void terminal_set_cursor(struct terminal *tm, const term_pos_t *pos);
214void terminal_set_winsize_blank(struct terminal *tm,
215 const struct winsize *size, int blank, const term_attr_t *attr);
216void terminal_set_winsize(struct terminal *tm, const struct winsize *size);
217void terminal_mute(struct terminal *tm, int yes);
218void terminal_input_char(struct terminal *tm, term_char_t c);
219void terminal_input_raw(struct terminal *tm, char c);
220void terminal_input_special(struct terminal *tm, unsigned int k);
221
222void termcn_cnregister(struct terminal *tm);
223
224/* Kernel console helper interface. */
225extern const struct consdev_ops termcn_cnops;
226
227#define TERMINAL_DECLARE_EARLY(name, class, softc) \
228 static struct terminal name = { \
229 .tm_class = &class, \
230 .tm_softc = softc, \
231 .tm_flags = TF_CONS, \
232 }; \
233 CONSOLE_DEVICE(name ## _consdev, termcn_cnops, &name)
234
235#endif /* _KERNEL */
236
237#endif /* !_SYS_TERMINAL_H_ */