/* Copyright 2019 The Chromium OS Authors. All rights reserved. * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "common.h" #include "config.h" #include "console.h" #include "timer.h" #include "uart.h" #include "usb_console.h" #include "util.h" #include "watchdog.h" #ifndef SECTION_IS_RO /* * Microseconds time to drain entire UART_TX console buffer at 115200 b/s, 10 * bits per character. */ #define BUFFER_DRAIN_TIME_US (1000000UL * 10 * CONFIG_UART_TX_BUF_SIZE \ / CONFIG_UART_BAUD_RATE) /* * Generate a stream of characters on the UART (and USB) console. * * The stream is an ever incrementing pattern of characters from the following * set: 0..9A..Za..z. * * The two optional integer command line arguments work as follows: * * argv[1] - reset the pattern after this many characters have been printed. * Setting this value to the width of the terminal window results * in a very regular stream showing on the terminal, where it is * easy to observe disruptions. * argv[2] - limit number of printed characters to this amount. If not * specified - keep printing indefinitely. * * Hitting 'x' on the keyboard stops the generator. */ static int command_chargen(int argc, char **argv) { int wrap_value = 0; int wrap_counter = 0; uint8_t c; uint32_t seq_counter = 0; uint32_t seq_number = 0; timestamp_t prev_watchdog_time; int (*putc_)(int c) = uart_putc; int (*tx_is_blocked_)(void) = uart_buffer_full; while (uart_getc() != -1 || usb_getc() != -1) ; /* Drain received characters, if any. */ if (argc > 1) wrap_value = atoi(argv[1]); if (argc > 2) seq_number = atoi(argv[2]); #if defined(CONFIG_USB_CONSOLE) || defined(CONFIG_USB_CONSOLE_STREAM) if (argc > 3) { if (memcmp(argv[3], "usb", 3)) return EC_ERROR_PARAM3; putc_ = usb_putc; tx_is_blocked_ = usb_console_tx_blocked; } #endif uart_shell_stop(); c = '0'; prev_watchdog_time = get_time(); while (uart_getc() != 'x' && usb_getc() != 'x') { timestamp_t current_time; while (tx_is_blocked_()) { /* * Let's let other tasks run for a bit while buffer is * being drained a little. */ usleep(BUFFER_DRAIN_TIME_US/10); current_time = get_time(); if ((current_time.val - prev_watchdog_time.val) < (CONFIG_WATCHDOG_PERIOD_MS * 1000 / 2)) continue; watchdog_reload(); prev_watchdog_time.val = current_time.val; } if (IS_ENABLED(CONFIG_ZEPHYR) && c == '0') watchdog_reload(); putc_(c++); if (seq_number && (++seq_counter == seq_number)) break; if (wrap_value && (++wrap_counter == wrap_value)) { c = '0'; wrap_counter = 0; continue; } if (c == ('z' + 1)) c = '0'; else if (c == ('Z' + 1)) c = 'a'; else if (c == ('9' + 1)) c = 'A'; } putc_('\n'); uart_shell_start(); return EC_SUCCESS; } DECLARE_SAFE_CONSOLE_COMMAND(chargen, command_chargen, #if defined(CONFIG_USB_CONSOLE) || defined(CONFIG_USB_CONSOLE_STREAM) "[seq_length [num_chars [usb]]]", #else "[seq_length [num_chars]]", #endif "Generate a constant stream of characters on the " "UART console,\nrepeating every 'seq_length' " "characters, up to 'num_chars' total." ); #endif /* !SECTION_IS_RO */