1 /* ----------------------------------------------------------------------- *
3 * Copyright 2004-2009 H. Peter Anvin - All Rights Reserved
4 * Copyright 2009 Intel Corporation; author: H. Peter Anvin
6 * Permission is hereby granted, free of charge, to any person
7 * obtaining a copy of this software and associated documentation
8 * files (the "Software"), to deal in the Software without
9 * restriction, including without limitation the rights to use,
10 * copy, modify, merge, publish, distribute, sublicense, and/or
11 * sell copies of the Software, and to permit persons to whom
12 * the Software is furnished to do so, subject to the following
15 * The above copyright notice and this permission notice shall
16 * be included in all copies or substantial portions of the Software.
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
19 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
20 * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
21 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
22 * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
23 * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
24 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
25 * OTHER DEALINGS IN THE SOFTWARE.
27 * ----------------------------------------------------------------------- */
32 * Write to the screen using ANSI control codes (about as capable as
40 #include <klibc/compiler.h>
41 #include <syslinux/config.h>
44 #include <syslinux/firmware.h>
47 static void ansicon_erase(const struct term_state
*, int, int, int, int);
48 static void ansicon_write_char(int, int, uint8_t, const struct term_state
*);
49 static void ansicon_showcursor(const struct term_state
*);
50 static void ansicon_scroll_up(const struct term_state
*);
51 static void ansicon_set_cursor(int, int, bool);
53 static struct term_state ts
;
54 struct ansi_ops __ansicon_ops
= {
55 .erase
= ansicon_erase
,
56 .write_char
= ansicon_write_char
,
57 .showcursor
= ansicon_showcursor
,
58 .set_cursor
= ansicon_set_cursor
,
59 .scroll_up
= ansicon_scroll_up
,
60 .beep
= __ansicon_beep
,
63 static struct term_info ti
= {
69 #define TEXT_MODE 0x0005
71 /* Reference counter to the screen, to keep track of if we need
73 static int ansicon_counter
= 0;
76 int __ansicon_open(struct file_info
*fp
)
78 if (!ansicon_counter
) {
79 /* Are we disabled? */
80 if (syslinux_serial_console_info()->flowctl
& 0x8000) {
86 firmware
->o_ops
->text_mode();
89 firmware
->o_ops
->get_mode(&ti
.cols
, &ti
.rows
);
92 /* Get cursor shape and position */
93 firmware
->o_ops
->get_cursor(&ti
.ts
->xy
.x
, &ti
.ts
->xy
.y
);
104 int __ansicon_close(struct file_info
*fp
)
112 /* Turn ANSI attributes into VGA attributes */
113 static uint8_t ansicon_attribute(const struct term_state
*st
)
120 else if (st
->intensity
== 0)
134 if (st
->intensity
== 2)
137 return (bg
<< 4) | fg
;
140 /* Erase a region of the screen */
141 static void ansicon_erase(const struct term_state
*st
,
142 int x0
, int y0
, int x1
, int y1
)
144 uint8_t attribute
= ansicon_attribute(st
);
146 if (firmware
->o_ops
->erase
)
147 firmware
->o_ops
->erase(x0
, y0
, x1
, y1
, attribute
);
150 /* Show or hide the cursor */
151 static void ansicon_showcursor(const struct term_state
*st
)
153 firmware
->o_ops
->showcursor(st
);
156 static void ansicon_set_cursor(int x
, int y
, bool visible
)
158 firmware
->o_ops
->set_cursor(x
, y
, visible
);
161 static void ansicon_write_char(int x
, int y
, uint8_t ch
,
162 const struct term_state
*st
)
164 uint8_t attribute
= ansicon_attribute(st
);
165 ansicon_set_cursor(x
, y
, false);
167 firmware
->o_ops
->write_char(ch
, attribute
);
170 static void ansicon_scroll_up(const struct term_state
*st
)
172 uint8_t rows
, cols
, attribute
;
176 attribute
= ansicon_attribute(st
);
178 firmware
->o_ops
->scroll_up(cols
, rows
, attribute
);
181 ssize_t
__ansicon_write(struct file_info
*fp
, const void *buf
, size_t count
)
183 const unsigned char *bufp
= buf
;
189 return count
; /* Nothing to do */
192 __ansi_putchar(&ti
, *bufp
++);
199 void __ansicon_beep(void)
201 if (firmware
->o_ops
->beep
)
202 firmware
->o_ops
->beep();
205 const struct output_dev dev_ansicon_w
= {
206 .dev_magic
= __DEV_MAGIC
,
207 .flags
= __DEV_TTY
| __DEV_OUTPUT
,
208 .fileflags
= O_WRONLY
| O_CREAT
| O_TRUNC
| O_APPEND
,
209 .write
= __ansicon_write
,
210 .close
= __ansicon_close
,
211 .open
= __ansicon_open
,