| // SPDX-License-Identifier: GPL-2.0-or-later |
| /* |
| * A udbg backend which logs messages and reads input from in memory |
| * buffers. |
| * |
| * The console output can be read from memcons_output which is a |
| * circular buffer whose next write position is stored in memcons.output_pos. |
| * |
| * Input may be passed by writing into the memcons_input buffer when it is |
| * empty. The input buffer is empty when both input_pos == input_start and |
| * *input_start == '\0'. |
| * |
| * Copyright (C) 2003-2005 Anton Blanchard and Milton Miller, IBM Corp |
| * Copyright (C) 2013 Alistair Popple, IBM Corp |
| */ |
| |
| #include <linux/kernel.h> |
| #include <asm/barrier.h> |
| #include <asm/page.h> |
| #include <asm/processor.h> |
| #include <asm/udbg.h> |
| |
| struct memcons { |
| char *output_start; |
| char *output_pos; |
| char *output_end; |
| char *input_start; |
| char *input_pos; |
| char *input_end; |
| }; |
| |
| static char memcons_output[CONFIG_PPC_MEMCONS_OUTPUT_SIZE]; |
| static char memcons_input[CONFIG_PPC_MEMCONS_INPUT_SIZE]; |
| |
| struct memcons memcons = { |
| .output_start = memcons_output, |
| .output_pos = memcons_output, |
| .output_end = &memcons_output[CONFIG_PPC_MEMCONS_OUTPUT_SIZE], |
| .input_start = memcons_input, |
| .input_pos = memcons_input, |
| .input_end = &memcons_input[CONFIG_PPC_MEMCONS_INPUT_SIZE], |
| }; |
| |
| void memcons_putc(char c) |
| { |
| char *new_output_pos; |
| |
| *memcons.output_pos = c; |
| wmb(); |
| new_output_pos = memcons.output_pos + 1; |
| if (new_output_pos >= memcons.output_end) |
| new_output_pos = memcons.output_start; |
| |
| memcons.output_pos = new_output_pos; |
| } |
| |
| int memcons_getc_poll(void) |
| { |
| char c; |
| char *new_input_pos; |
| |
| if (*memcons.input_pos) { |
| c = *memcons.input_pos; |
| |
| new_input_pos = memcons.input_pos + 1; |
| if (new_input_pos >= memcons.input_end) |
| new_input_pos = memcons.input_start; |
| else if (*new_input_pos == '\0') |
| new_input_pos = memcons.input_start; |
| |
| *memcons.input_pos = '\0'; |
| wmb(); |
| memcons.input_pos = new_input_pos; |
| return c; |
| } |
| |
| return -1; |
| } |
| |
| int memcons_getc(void) |
| { |
| int c; |
| |
| while (1) { |
| c = memcons_getc_poll(); |
| if (c == -1) |
| cpu_relax(); |
| else |
| break; |
| } |
| |
| return c; |
| } |
| |
| void __init udbg_init_memcons(void) |
| { |
| udbg_putc = memcons_putc; |
| udbg_getc = memcons_getc; |
| udbg_getc_poll = memcons_getc_poll; |
| } |