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> 14*7d95f2a3SSimon Glass #include <lcd.h> 157accb6eaSSimon Glass #include <os.h> 16cef46b77SMarek Vasut #include <serial.h> 17cef46b77SMarek Vasut #include <linux/compiler.h> 187accb6eaSSimon Glass 19e101550aSTaylor Hutt /* 20e101550aSTaylor Hutt * 21e101550aSTaylor Hutt * serial_buf: A buffer that holds keyboard characters for the 22e101550aSTaylor Hutt * Sandbox U-boot. 23e101550aSTaylor Hutt * 24e101550aSTaylor Hutt * invariants: 25e101550aSTaylor Hutt * serial_buf_write == serial_buf_read -> empty buffer 26e101550aSTaylor Hutt * (serial_buf_write + 1) % 16 == serial_buf_read -> full buffer 27e101550aSTaylor Hutt */ 28e101550aSTaylor Hutt static char serial_buf[16]; 29e101550aSTaylor Hutt static unsigned int serial_buf_write; 30e101550aSTaylor Hutt static unsigned int serial_buf_read; 31e101550aSTaylor Hutt 32cef46b77SMarek Vasut static int sandbox_serial_init(void) 337accb6eaSSimon Glass { 34ab06a758SMike Frysinger os_tty_raw(0); 357accb6eaSSimon Glass return 0; 367accb6eaSSimon Glass } 377accb6eaSSimon Glass 38cef46b77SMarek Vasut static void sandbox_serial_setbrg(void) 397accb6eaSSimon Glass { 407accb6eaSSimon Glass } 417accb6eaSSimon Glass 42cef46b77SMarek Vasut static void sandbox_serial_putc(const char ch) 437accb6eaSSimon Glass { 447accb6eaSSimon Glass os_write(1, &ch, 1); 457accb6eaSSimon Glass } 467accb6eaSSimon Glass 47cef46b77SMarek Vasut static void sandbox_serial_puts(const char *str) 487accb6eaSSimon Glass { 495778d54aSMike Frysinger os_write(1, str, strlen(str)); 507accb6eaSSimon Glass } 517accb6eaSSimon Glass 52e101550aSTaylor Hutt static unsigned int increment_buffer_index(unsigned int index) 537accb6eaSSimon Glass { 54e101550aSTaylor Hutt return (index + 1) % ARRAY_SIZE(serial_buf); 557accb6eaSSimon Glass } 567accb6eaSSimon Glass 57cef46b77SMarek Vasut static int sandbox_serial_tstc(void) 587accb6eaSSimon Glass { 59e101550aSTaylor Hutt const unsigned int next_index = 60e101550aSTaylor Hutt increment_buffer_index(serial_buf_write); 61e101550aSTaylor Hutt ssize_t count; 62e101550aSTaylor Hutt 63e101550aSTaylor Hutt os_usleep(100); 64*7d95f2a3SSimon Glass #ifdef CONFIG_LCD 65*7d95f2a3SSimon Glass lcd_sync(); 66*7d95f2a3SSimon Glass #endif 67e101550aSTaylor Hutt if (next_index == serial_buf_read) 68e101550aSTaylor Hutt return 1; /* buffer full */ 69e101550aSTaylor Hutt 70e101550aSTaylor Hutt count = os_read_no_block(0, &serial_buf[serial_buf_write], 1); 71e101550aSTaylor Hutt if (count == 1) 72e101550aSTaylor Hutt serial_buf_write = next_index; 73e101550aSTaylor Hutt return serial_buf_write != serial_buf_read; 74e101550aSTaylor Hutt } 75e101550aSTaylor Hutt 76e101550aSTaylor Hutt static int sandbox_serial_getc(void) 77e101550aSTaylor Hutt { 78e101550aSTaylor Hutt int result; 79e101550aSTaylor Hutt 80e101550aSTaylor Hutt while (!sandbox_serial_tstc()) 81e101550aSTaylor Hutt ; /* buffer empty */ 82e101550aSTaylor Hutt 83e101550aSTaylor Hutt result = serial_buf[serial_buf_read]; 84e101550aSTaylor Hutt serial_buf_read = increment_buffer_index(serial_buf_read); 85e101550aSTaylor Hutt return result; 867accb6eaSSimon Glass } 87cef46b77SMarek Vasut 88cef46b77SMarek Vasut static struct serial_device sandbox_serial_drv = { 89cef46b77SMarek Vasut .name = "sandbox_serial", 90cef46b77SMarek Vasut .start = sandbox_serial_init, 91cef46b77SMarek Vasut .stop = NULL, 92cef46b77SMarek Vasut .setbrg = sandbox_serial_setbrg, 93cef46b77SMarek Vasut .putc = sandbox_serial_putc, 94cef46b77SMarek Vasut .puts = sandbox_serial_puts, 95cef46b77SMarek Vasut .getc = sandbox_serial_getc, 96cef46b77SMarek Vasut .tstc = sandbox_serial_tstc, 97cef46b77SMarek Vasut }; 98cef46b77SMarek Vasut 99cef46b77SMarek Vasut void sandbox_serial_initialize(void) 100cef46b77SMarek Vasut { 101cef46b77SMarek Vasut serial_register(&sandbox_serial_drv); 102cef46b77SMarek Vasut } 103cef46b77SMarek Vasut 104cef46b77SMarek Vasut __weak struct serial_device *default_serial_console(void) 105cef46b77SMarek Vasut { 106cef46b77SMarek Vasut return &sandbox_serial_drv; 107cef46b77SMarek Vasut } 108