17accb6eaSSimon Glass /* 27accb6eaSSimon Glass * Copyright (c) 2011 The Chromium OS Authors. 37accb6eaSSimon Glass * 41a459660SWolfgang Denk * SPDX-License-Identifier: GPL-2.0+ 57accb6eaSSimon Glass */ 67accb6eaSSimon Glass 77accb6eaSSimon Glass /* 87accb6eaSSimon Glass * This provide a test serial port. It provides an emulated serial port where 97accb6eaSSimon Glass * a test program and read out the serial output and inject serial input for 107accb6eaSSimon Glass * U-Boot. 117accb6eaSSimon Glass */ 127accb6eaSSimon Glass 137accb6eaSSimon Glass #include <common.h> 147d95f2a3SSimon Glass #include <lcd.h> 157accb6eaSSimon Glass #include <os.h> 16cef46b77SMarek Vasut #include <serial.h> 17cef46b77SMarek Vasut #include <linux/compiler.h> 18*ffb87905SSimon Glass #include <asm/state.h> 197accb6eaSSimon Glass 20e101550aSTaylor Hutt /* 21e101550aSTaylor Hutt * 22e101550aSTaylor Hutt * serial_buf: A buffer that holds keyboard characters for the 23e101550aSTaylor Hutt * Sandbox U-boot. 24e101550aSTaylor Hutt * 25e101550aSTaylor Hutt * invariants: 26e101550aSTaylor Hutt * serial_buf_write == serial_buf_read -> empty buffer 27e101550aSTaylor Hutt * (serial_buf_write + 1) % 16 == serial_buf_read -> full buffer 28e101550aSTaylor Hutt */ 29e101550aSTaylor Hutt static char serial_buf[16]; 30e101550aSTaylor Hutt static unsigned int serial_buf_write; 31e101550aSTaylor Hutt static unsigned int serial_buf_read; 32e101550aSTaylor Hutt 33cef46b77SMarek Vasut static int sandbox_serial_init(void) 347accb6eaSSimon Glass { 35*ffb87905SSimon Glass struct sandbox_state *state = state_get_current(); 36*ffb87905SSimon Glass 37*ffb87905SSimon Glass if (state->term_raw != STATE_TERM_COOKED) 38*ffb87905SSimon Glass os_tty_raw(0, state->term_raw == STATE_TERM_RAW_WITH_SIGS); 397accb6eaSSimon Glass return 0; 407accb6eaSSimon Glass } 417accb6eaSSimon Glass 42cef46b77SMarek Vasut static void sandbox_serial_setbrg(void) 437accb6eaSSimon Glass { 447accb6eaSSimon Glass } 457accb6eaSSimon Glass 46cef46b77SMarek Vasut static void sandbox_serial_putc(const char ch) 477accb6eaSSimon Glass { 487accb6eaSSimon Glass os_write(1, &ch, 1); 497accb6eaSSimon Glass } 507accb6eaSSimon Glass 51cef46b77SMarek Vasut static void sandbox_serial_puts(const char *str) 527accb6eaSSimon Glass { 535778d54aSMike Frysinger os_write(1, str, strlen(str)); 547accb6eaSSimon Glass } 557accb6eaSSimon Glass 56e101550aSTaylor Hutt static unsigned int increment_buffer_index(unsigned int index) 577accb6eaSSimon Glass { 58e101550aSTaylor Hutt return (index + 1) % ARRAY_SIZE(serial_buf); 597accb6eaSSimon Glass } 607accb6eaSSimon Glass 61cef46b77SMarek Vasut static int sandbox_serial_tstc(void) 627accb6eaSSimon Glass { 63e101550aSTaylor Hutt const unsigned int next_index = 64e101550aSTaylor Hutt increment_buffer_index(serial_buf_write); 65e101550aSTaylor Hutt ssize_t count; 66e101550aSTaylor Hutt 67e101550aSTaylor Hutt os_usleep(100); 687d95f2a3SSimon Glass #ifdef CONFIG_LCD 697d95f2a3SSimon Glass lcd_sync(); 707d95f2a3SSimon Glass #endif 71e101550aSTaylor Hutt if (next_index == serial_buf_read) 72e101550aSTaylor Hutt return 1; /* buffer full */ 73e101550aSTaylor Hutt 74e101550aSTaylor Hutt count = os_read_no_block(0, &serial_buf[serial_buf_write], 1); 75e101550aSTaylor Hutt if (count == 1) 76e101550aSTaylor Hutt serial_buf_write = next_index; 77e101550aSTaylor Hutt return serial_buf_write != serial_buf_read; 78e101550aSTaylor Hutt } 79e101550aSTaylor Hutt 80e101550aSTaylor Hutt static int sandbox_serial_getc(void) 81e101550aSTaylor Hutt { 82e101550aSTaylor Hutt int result; 83e101550aSTaylor Hutt 84e101550aSTaylor Hutt while (!sandbox_serial_tstc()) 85e101550aSTaylor Hutt ; /* buffer empty */ 86e101550aSTaylor Hutt 87e101550aSTaylor Hutt result = serial_buf[serial_buf_read]; 88e101550aSTaylor Hutt serial_buf_read = increment_buffer_index(serial_buf_read); 89e101550aSTaylor Hutt return result; 907accb6eaSSimon Glass } 91cef46b77SMarek Vasut 92cef46b77SMarek Vasut static struct serial_device sandbox_serial_drv = { 93cef46b77SMarek Vasut .name = "sandbox_serial", 94cef46b77SMarek Vasut .start = sandbox_serial_init, 95cef46b77SMarek Vasut .stop = NULL, 96cef46b77SMarek Vasut .setbrg = sandbox_serial_setbrg, 97cef46b77SMarek Vasut .putc = sandbox_serial_putc, 98cef46b77SMarek Vasut .puts = sandbox_serial_puts, 99cef46b77SMarek Vasut .getc = sandbox_serial_getc, 100cef46b77SMarek Vasut .tstc = sandbox_serial_tstc, 101cef46b77SMarek Vasut }; 102cef46b77SMarek Vasut 103cef46b77SMarek Vasut void sandbox_serial_initialize(void) 104cef46b77SMarek Vasut { 105cef46b77SMarek Vasut serial_register(&sandbox_serial_drv); 106cef46b77SMarek Vasut } 107cef46b77SMarek Vasut 108cef46b77SMarek Vasut __weak struct serial_device *default_serial_console(void) 109cef46b77SMarek Vasut { 110cef46b77SMarek Vasut return &sandbox_serial_drv; 111cef46b77SMarek Vasut } 112