aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/misc/cbmem_console.c
blob: 8220addd579bbc147c5ea138fe9324a8a3637e72 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
// SPDX-License-Identifier: GPL-2.0
/*
 * Copyright (C) 2011 The ChromiumOS Authors.  All rights reserved.
 */

#include <console.h>
#include <linux/string.h>
#include <asm/cb_sysinfo.h>

void cbmemc_putc(struct stdio_dev *dev, char data)
{
	const struct sysinfo_t *sysinfo = cb_get_sysinfo();
	struct cbmem_console *cons;
	uint pos, flags;

	if (!sysinfo)
		return;
	cons = sysinfo->cbmem_cons;
	if (!cons)
		return;

	pos = cons->cursor & CBMC_CURSOR_MASK;

	/* preserve the overflow flag if present */
	flags = cons->cursor & ~CBMC_CURSOR_MASK;

	cons->body[pos++] = data;

	/*
	 * Deal with overflow - the flag may be cleared by another program which
	 * reads the buffer out later, e.g. Linux
	 */
	if (pos >= cons->size) {
		pos = 0;
		flags |= CBMC_OVERFLOW;
	}

	cons->cursor = flags | pos;
}

void cbmemc_puts(struct stdio_dev *dev, const char *str)
{
	char c;

	while ((c = *str++) != 0)
		cbmemc_putc(dev, c);
}

int cbmemc_init(void)
{
	int rc;
	struct stdio_dev cons_dev;

	memset(&cons_dev, 0, sizeof(cons_dev));

	strcpy(cons_dev.name, "cbmem");
	cons_dev.flags = DEV_FLAGS_OUTPUT; /* Output only */
	cons_dev.putc  = cbmemc_putc;
	cons_dev.puts  = cbmemc_puts;

	rc = stdio_register(&cons_dev);

	return (rc == 0) ? 1 : rc;
}