On Mon, 2013-04-29 at 16:24 +1000, Alistair Popple wrote:
> This patch adds a new udbg early debug console which utilises
> statically defined input and output buffers stored within the kernel
> BSS. It is primarily designed to assist with bring up of new hardware
> which may not have a working console but which has a method of
> reading/writing kernel memory.

This probably need some memory barriers :-)

Cheers,
Ben.

> Signed-off-by: Alistair Popple <alist...@popple.id.au>
> ---
>  arch/powerpc/Kconfig.debug         |   23 ++++++++++
>  arch/powerpc/include/asm/udbg.h    |    1 +
>  arch/powerpc/kernel/udbg.c         |    3 ++
>  arch/powerpc/sysdev/Makefile       |    2 +
>  arch/powerpc/sysdev/udbg_memcons.c |   85 
> ++++++++++++++++++++++++++++++++++++
>  5 files changed, 114 insertions(+)
>  create mode 100644 arch/powerpc/sysdev/udbg_memcons.c
> 
> diff --git a/arch/powerpc/Kconfig.debug b/arch/powerpc/Kconfig.debug
> index 5416e28..863d877 100644
> --- a/arch/powerpc/Kconfig.debug
> +++ b/arch/powerpc/Kconfig.debug
> @@ -262,8 +262,31 @@ config PPC_EARLY_DEBUG_OPAL_HVSI
>         Select this to enable early debugging for the PowerNV platform
>         using an "hvsi" console
>  
> +config PPC_EARLY_DEBUG_MEMCONS
> +     bool "In memory console"
> +     help
> +       Select this to enable early debugging using an in memory console.
> +       This console provides input and output buffers stored within the
> +       kernel BSS and should be safe to select on any system. A debugger
> +       can then be used to read kernel output or send input to the console.
>  endchoice
>  
> +config PPC_MEMCONS_OUTPUT_SIZE
> +     int "In memory console output buffer size"
> +     depends on PPC_EARLY_DEBUG_MEMCONS
> +     default 4096
> +     help
> +       Selects the size of the output buffer (in bytes) of the in memory
> +       console.
> +
> +config PPC_MEMCONS_INPUT_SIZE
> +     int "In memory console input buffer size"
> +     depends on PPC_EARLY_DEBUG_MEMCONS
> +     default 128
> +     help
> +       Selects the size of the input buffer (in bytes) of the in memory
> +       console.
> +
>  config PPC_EARLY_DEBUG_OPAL
>       def_bool y
>       depends on PPC_EARLY_DEBUG_OPAL_RAW || PPC_EARLY_DEBUG_OPAL_HVSI
> diff --git a/arch/powerpc/include/asm/udbg.h b/arch/powerpc/include/asm/udbg.h
> index 5a7510e..dc59091 100644
> --- a/arch/powerpc/include/asm/udbg.h
> +++ b/arch/powerpc/include/asm/udbg.h
> @@ -52,6 +52,7 @@ extern void __init udbg_init_40x_realmode(void);
>  extern void __init udbg_init_cpm(void);
>  extern void __init udbg_init_usbgecko(void);
>  extern void __init udbg_init_wsp(void);
> +extern void __init udbg_init_memcons(void);
>  extern void __init udbg_init_ehv_bc(void);
>  extern void __init udbg_init_ps3gelic(void);
>  extern void __init udbg_init_debug_opal_raw(void);
> diff --git a/arch/powerpc/kernel/udbg.c b/arch/powerpc/kernel/udbg.c
> index f974849..efa5c93 100644
> --- a/arch/powerpc/kernel/udbg.c
> +++ b/arch/powerpc/kernel/udbg.c
> @@ -64,6 +64,9 @@ void __init udbg_early_init(void)
>       udbg_init_usbgecko();
>  #elif defined(CONFIG_PPC_EARLY_DEBUG_WSP)
>       udbg_init_wsp();
> +#elif defined(CONFIG_PPC_EARLY_DEBUG_MEMCONS)
> +     /* In memory console */
> +     udbg_init_memcons();
>  #elif defined(CONFIG_PPC_EARLY_DEBUG_EHV_BC)
>       udbg_init_ehv_bc();
>  #elif defined(CONFIG_PPC_EARLY_DEBUG_PS3GELIC)
> diff --git a/arch/powerpc/sysdev/Makefile b/arch/powerpc/sysdev/Makefile
> index b0a518e..99464a7 100644
> --- a/arch/powerpc/sysdev/Makefile
> +++ b/arch/powerpc/sysdev/Makefile
> @@ -64,6 +64,8 @@ endif
>  
>  obj-$(CONFIG_PPC_SCOM)               += scom.o
>  
> +obj-$(CONFIG_PPC_EARLY_DEBUG_MEMCONS)        += udbg_memcons.o
> +
>  subdir-ccflags-$(CONFIG_PPC_WERROR) := -Werror
>  
>  obj-$(CONFIG_PPC_XICS)               += xics/
> diff --git a/arch/powerpc/sysdev/udbg_memcons.c 
> b/arch/powerpc/sysdev/udbg_memcons.c
> new file mode 100644
> index 0000000..36f5c44
> --- /dev/null
> +++ b/arch/powerpc/sysdev/udbg_memcons.c
> @@ -0,0 +1,85 @@
> +/*
> + * A udbg backend which logs messages and reads input from in memory
> + * buffers.
> + *
> + * Copyright (C) 2003-2005 Anton Blanchard and Milton Miller, IBM Corp
> + * Copyright (C) 2013 Alistair Popple, IBM Corp
> + *
> + *      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.
> + */
> +
> +#include <linux/init.h>
> +#include <linux/kernel.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)
> +{
> +     *memcons.output_pos++ = c;
> +     if (memcons.output_pos >= memcons.output_end)
> +             memcons.output_pos = memcons.output_start;
> +}
> +
> +int memcons_getc_poll(void)
> +{
> +     char c;
> +
> +     if (*memcons.input_pos) {
> +             c = *memcons.input_pos;
> +             *memcons.input_pos++ = 0;
> +
> +             if (*memcons.input_pos == '\0' || memcons.input_pos >= 
> memcons.input_end)
> +                     memcons.input_pos = memcons.input_start;
> +
> +             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;
> +}


_______________________________________________
Linuxppc-dev mailing list
Linuxppc-dev@lists.ozlabs.org
https://lists.ozlabs.org/listinfo/linuxppc-dev

Reply via email to