2013-01-08 22:40:58 +08:00
|
|
|
/*
|
2021-02-19 23:55:17 +08:00
|
|
|
* Copyright (c) 2006-2021, RT-Thread Development Team
|
2013-01-08 22:40:58 +08:00
|
|
|
*
|
2021-02-19 23:55:17 +08:00
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
2013-01-08 22:40:58 +08:00
|
|
|
*
|
|
|
|
* Change Logs:
|
|
|
|
* Date Author Notes
|
|
|
|
* 2006-09-15 QiuYi the first version
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <rtthread.h>
|
|
|
|
#include <rthw.h>
|
|
|
|
|
|
|
|
#include <bsp.h>
|
|
|
|
|
|
|
|
static unsigned addr_6845;
|
|
|
|
static rt_uint16_t *crt_buf;
|
|
|
|
static rt_int16_t crt_pos;
|
|
|
|
|
2017-08-18 11:38:00 +08:00
|
|
|
extern void init_keyboard();
|
|
|
|
extern void rt_keyboard_isr(void);
|
|
|
|
extern rt_bool_t rt_keyboard_getc(char* c);
|
2013-01-08 22:40:58 +08:00
|
|
|
|
2017-08-18 12:02:43 +08:00
|
|
|
extern void rt_serial_init(void);
|
|
|
|
extern char rt_serial_getc(void);
|
|
|
|
extern void rt_serial_putc(const char c);
|
|
|
|
|
2017-08-20 20:24:07 +08:00
|
|
|
void rt_console_putc(int c);
|
2013-01-08 22:40:58 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @addtogroup QEMU
|
|
|
|
*/
|
|
|
|
/*@{*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This function initializes cga
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void rt_cga_init(void)
|
|
|
|
{
|
|
|
|
rt_uint16_t volatile *cp;
|
|
|
|
rt_uint16_t was;
|
|
|
|
rt_uint32_t pos;
|
|
|
|
|
|
|
|
cp = (rt_uint16_t *) (CGA_BUF);
|
|
|
|
was = *cp;
|
|
|
|
*cp = (rt_uint16_t) 0xA55A;
|
|
|
|
if (*cp != 0xA55A)
|
|
|
|
{
|
|
|
|
cp = (rt_uint16_t *) (MONO_BUF);
|
|
|
|
addr_6845 = MONO_BASE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*cp = was;
|
|
|
|
addr_6845 = CGA_BASE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Extract cursor location */
|
|
|
|
outb(addr_6845, 14);
|
|
|
|
pos = inb(addr_6845+1) << 8;
|
|
|
|
outb(addr_6845, 15);
|
|
|
|
pos |= inb(addr_6845+1);
|
|
|
|
|
|
|
|
crt_buf = (rt_uint16_t *)cp;
|
|
|
|
crt_pos = pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This function will write a character to cga
|
|
|
|
*
|
|
|
|
* @param c the char to write
|
|
|
|
*/
|
|
|
|
static void rt_cga_putc(int c)
|
|
|
|
{
|
|
|
|
/* if no attribute given, then use black on white */
|
|
|
|
if (!(c & ~0xff)) c |= 0x0700;
|
|
|
|
|
|
|
|
switch (c & 0xff)
|
|
|
|
{
|
|
|
|
case '\b':
|
|
|
|
if (crt_pos > 0)
|
|
|
|
{
|
|
|
|
crt_pos--;
|
|
|
|
crt_buf[crt_pos] = (c&~0xff) | ' ';
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '\n':
|
|
|
|
crt_pos += CRT_COLS;
|
|
|
|
/* cascade */
|
|
|
|
case '\r':
|
|
|
|
crt_pos -= (crt_pos % CRT_COLS);
|
|
|
|
break;
|
|
|
|
case '\t':
|
|
|
|
rt_console_putc(' ');
|
|
|
|
rt_console_putc(' ');
|
|
|
|
rt_console_putc(' ');
|
|
|
|
rt_console_putc(' ');
|
|
|
|
rt_console_putc(' ');
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
crt_buf[crt_pos++] = c; /* write the character */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (crt_pos >= CRT_SIZE)
|
|
|
|
{
|
|
|
|
rt_int32_t i;
|
|
|
|
rt_memcpy(crt_buf, crt_buf + CRT_COLS, (CRT_SIZE - CRT_COLS) << 1);
|
|
|
|
for (i = CRT_SIZE - CRT_COLS; i < CRT_SIZE; i++)
|
|
|
|
crt_buf[i] = 0x0700 | ' ';
|
|
|
|
crt_pos -= CRT_COLS;
|
|
|
|
}
|
|
|
|
|
|
|
|
outb(addr_6845, 14);
|
|
|
|
outb(addr_6845+1, crt_pos >> 8);
|
|
|
|
outb(addr_6845, 15);
|
|
|
|
outb(addr_6845+1, crt_pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This function will write a character to serial an cga
|
|
|
|
*
|
|
|
|
* @param c the char to write
|
|
|
|
*/
|
2017-08-20 20:24:07 +08:00
|
|
|
void rt_console_putc(int c)
|
2013-01-08 22:40:58 +08:00
|
|
|
{
|
|
|
|
rt_cga_putc(c);
|
2017-08-18 12:02:43 +08:00
|
|
|
rt_serial_putc(c);
|
2013-01-08 22:40:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* RT-Thread Device Interface */
|
|
|
|
#define CONSOLE_RX_BUFFER_SIZE 64
|
|
|
|
static struct rt_device console_device;
|
|
|
|
static rt_uint8_t rx_buffer[CONSOLE_RX_BUFFER_SIZE];
|
|
|
|
static rt_uint32_t read_index, save_index;
|
|
|
|
|
|
|
|
static rt_err_t rt_console_init (rt_device_t dev)
|
|
|
|
{
|
|
|
|
return RT_EOK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static rt_err_t rt_console_open(rt_device_t dev, rt_uint16_t oflag)
|
|
|
|
{
|
|
|
|
return RT_EOK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static rt_err_t rt_console_close(rt_device_t dev)
|
|
|
|
{
|
|
|
|
return RT_EOK;
|
|
|
|
}
|
|
|
|
|
2017-10-16 13:23:03 +08:00
|
|
|
static rt_err_t rt_console_control(rt_device_t dev, int cmd, void *args)
|
2013-01-08 22:40:58 +08:00
|
|
|
{
|
|
|
|
return RT_EOK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static rt_size_t rt_console_write(rt_device_t dev, rt_off_t pos, const void * buffer, rt_size_t size)
|
|
|
|
{
|
|
|
|
rt_size_t i = size;
|
|
|
|
const char* str = buffer;
|
|
|
|
|
|
|
|
while(i--)
|
|
|
|
{
|
|
|
|
rt_console_putc(*str++);
|
|
|
|
}
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
static rt_size_t rt_console_read(rt_device_t dev, rt_off_t pos, void* buffer, rt_size_t size)
|
|
|
|
{
|
|
|
|
rt_uint8_t* ptr = buffer;
|
|
|
|
rt_err_t err_code = RT_EOK;
|
|
|
|
|
|
|
|
/* interrupt mode Rx */
|
|
|
|
while (size)
|
|
|
|
{
|
|
|
|
rt_base_t level;
|
|
|
|
|
|
|
|
/* disable interrupt */
|
|
|
|
level = rt_hw_interrupt_disable();
|
|
|
|
|
|
|
|
if (read_index != save_index)
|
|
|
|
{
|
|
|
|
/* read a character */
|
|
|
|
*ptr++ = rx_buffer[read_index];
|
|
|
|
size--;
|
|
|
|
|
|
|
|
/* move to next position */
|
|
|
|
read_index ++;
|
|
|
|
if (read_index >= CONSOLE_RX_BUFFER_SIZE)
|
|
|
|
read_index = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* set error code */
|
|
|
|
err_code = -RT_EEMPTY;
|
|
|
|
|
|
|
|
/* enable interrupt */
|
|
|
|
rt_hw_interrupt_enable(level);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* enable interrupt */
|
|
|
|
rt_hw_interrupt_enable(level);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set error code */
|
|
|
|
rt_set_errno(err_code);
|
|
|
|
return (rt_uint32_t)ptr - (rt_uint32_t)buffer;
|
|
|
|
}
|
|
|
|
|
2015-09-15 19:50:29 +08:00
|
|
|
static void rt_console_isr(int vector, void* param)
|
2013-01-08 22:40:58 +08:00
|
|
|
{
|
|
|
|
char c;
|
2021-02-19 23:55:17 +08:00
|
|
|
rt_bool_t ret;
|
2013-01-08 22:40:58 +08:00
|
|
|
rt_base_t level;
|
|
|
|
|
2021-02-19 23:55:17 +08:00
|
|
|
if(INTUART0_RX == vector)
|
|
|
|
{
|
|
|
|
c = rt_serial_getc();
|
|
|
|
ret = RT_TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
rt_keyboard_isr();
|
|
|
|
ret = rt_keyboard_getc(&c);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(ret == RT_FALSE)
|
|
|
|
{
|
|
|
|
/* do nothing */
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* disable interrupt */
|
|
|
|
level = rt_hw_interrupt_disable();
|
|
|
|
|
|
|
|
/* save character */
|
|
|
|
rx_buffer[save_index] = c;
|
|
|
|
save_index ++;
|
|
|
|
if (save_index >= CONSOLE_RX_BUFFER_SIZE)
|
|
|
|
save_index = 0;
|
|
|
|
|
|
|
|
/* if the next position is read index, discard this 'read char' */
|
|
|
|
if (save_index == read_index)
|
|
|
|
{
|
|
|
|
read_index ++;
|
|
|
|
if (read_index >= CONSOLE_RX_BUFFER_SIZE)
|
|
|
|
read_index = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* enable interrupt */
|
|
|
|
rt_hw_interrupt_enable(level);
|
|
|
|
}
|
2013-01-08 22:40:58 +08:00
|
|
|
|
|
|
|
/* invoke callback */
|
|
|
|
if (console_device.rx_indicate != RT_NULL)
|
|
|
|
{
|
|
|
|
rt_size_t rx_length;
|
|
|
|
|
|
|
|
/* get rx length */
|
|
|
|
rx_length = read_index > save_index ?
|
2021-02-19 23:55:17 +08:00
|
|
|
CONSOLE_RX_BUFFER_SIZE - read_index + save_index :
|
|
|
|
save_index - read_index;
|
2013-01-08 22:40:58 +08:00
|
|
|
|
|
|
|
if(rx_length > 0)
|
|
|
|
{
|
|
|
|
console_device.rx_indicate(&console_device, rx_length);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This function initializes console
|
|
|
|
*
|
|
|
|
*/
|
2021-02-19 23:55:17 +08:00
|
|
|
int rt_hw_console_init(void)
|
2013-01-08 22:40:58 +08:00
|
|
|
{
|
|
|
|
rt_cga_init();
|
2021-02-19 23:55:17 +08:00
|
|
|
rt_serial_init();
|
|
|
|
init_keyboard();
|
2013-01-08 22:40:58 +08:00
|
|
|
|
|
|
|
/* install keyboard isr */
|
2015-09-15 19:50:29 +08:00
|
|
|
rt_hw_interrupt_install(INTKEYBOARD, rt_console_isr, RT_NULL, "kbd");
|
2013-01-08 22:40:58 +08:00
|
|
|
rt_hw_interrupt_umask(INTKEYBOARD);
|
|
|
|
|
2017-08-18 12:02:43 +08:00
|
|
|
rt_hw_interrupt_install(INTUART0_RX, rt_console_isr, RT_NULL, "COM1");
|
|
|
|
rt_hw_interrupt_umask(INTUART0_RX);
|
|
|
|
|
2013-01-08 22:40:58 +08:00
|
|
|
console_device.type = RT_Device_Class_Char;
|
|
|
|
console_device.rx_indicate = RT_NULL;
|
|
|
|
console_device.tx_complete = RT_NULL;
|
|
|
|
console_device.init = rt_console_init;
|
|
|
|
console_device.open = rt_console_open;
|
|
|
|
console_device.close = rt_console_close;
|
|
|
|
console_device.read = rt_console_read;
|
|
|
|
console_device.write = rt_console_write;
|
|
|
|
console_device.control = rt_console_control;
|
|
|
|
console_device.user_data = RT_NULL;
|
|
|
|
|
|
|
|
/* register a character device */
|
|
|
|
rt_device_register(&console_device,
|
2021-02-19 23:55:17 +08:00
|
|
|
"console",
|
|
|
|
RT_DEVICE_FLAG_RDWR | RT_DEVICE_FLAG_INT_RX | RT_DEVICE_FLAG_STREAM);
|
|
|
|
|
|
|
|
return 0;
|
2013-01-08 22:40:58 +08:00
|
|
|
}
|
2021-02-19 23:55:17 +08:00
|
|
|
INIT_DEVICE_EXPORT(rt_hw_console_init);
|
2013-01-08 22:40:58 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This function is used to display a string on console, normally, it's
|
|
|
|
* invoked by rt_kprintf
|
|
|
|
*
|
|
|
|
* @param str the displayed string
|
|
|
|
*
|
|
|
|
* Modified:
|
|
|
|
* caoxl 2009-10-14
|
|
|
|
* the name is change to rt_hw_console_output in the v0.3.0
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void rt_hw_console_output(const char* str)
|
|
|
|
{
|
|
|
|
while (*str)
|
|
|
|
{
|
|
|
|
rt_console_putc (*str++);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*@}*/
|