2874c5fd28
Based on 1 normalized pattern(s): this program is free software you can redistribute it and or modify it under the terms of the gnu general public license as published by the free software foundation either version 2 of the license or at your option any later version extracted by the scancode license scanner the SPDX license identifier GPL-2.0-or-later has been chosen to replace the boilerplate/reference in 3029 file(s). Signed-off-by: Thomas Gleixner <tglx@linutronix.de> Reviewed-by: Allison Randal <allison@lohutok.net> Cc: linux-spdx@vger.kernel.org Link: https://lkml.kernel.org/r/20190527070032.746973796@linutronix.de Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
101 lines
2.1 KiB
C
101 lines
2.1 KiB
C
// 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 udbg_init_memcons(void)
|
|
{
|
|
udbg_putc = memcons_putc;
|
|
udbg_getc = memcons_getc;
|
|
udbg_getc_poll = memcons_getc_poll;
|
|
}
|