2012-05-30 18:13:09 +00:00
|
|
|
/*
|
2014-01-20 17:24:57 +00:00
|
|
|
* \brief Print to the standard output of the kernel
|
2012-05-30 18:13:09 +00:00
|
|
|
* \author Martin stein
|
|
|
|
* \date 2012-04-05
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2013-01-10 20:44:47 +00:00
|
|
|
* Copyright (C) 2012-2013 Genode Labs GmbH
|
2012-05-30 18:13:09 +00:00
|
|
|
*
|
|
|
|
* This file is part of the Genode OS framework, which is distributed
|
|
|
|
* under the terms of the GNU General Public License version 2.
|
|
|
|
*/
|
|
|
|
|
2013-11-14 16:29:34 +00:00
|
|
|
#ifndef _KERNEL__LOG_H_
|
|
|
|
#define _KERNEL__LOG_H_
|
2012-05-30 18:13:09 +00:00
|
|
|
|
|
|
|
/* Genode includes */
|
2013-11-14 16:29:34 +00:00
|
|
|
#include <kernel/interface.h>
|
2012-05-30 18:13:09 +00:00
|
|
|
|
2014-01-20 17:24:57 +00:00
|
|
|
namespace Kernel
|
2012-05-30 18:13:09 +00:00
|
|
|
{
|
|
|
|
/**
|
2014-01-20 17:24:57 +00:00
|
|
|
* Prints incoming streams to the standard output of the kernel
|
2012-05-30 18:13:09 +00:00
|
|
|
*/
|
2014-01-20 17:24:57 +00:00
|
|
|
class Log
|
2012-05-30 18:13:09 +00:00
|
|
|
{
|
2014-01-20 17:24:57 +00:00
|
|
|
private:
|
2012-05-30 18:13:09 +00:00
|
|
|
|
2014-01-20 17:24:57 +00:00
|
|
|
/**
|
|
|
|
* Print an unsigned 4 bit integer x as hexadecimal value
|
|
|
|
*/
|
|
|
|
void _print_4bit_hex(unsigned char x) const
|
|
|
|
{
|
|
|
|
/* decode to ASCII char */
|
|
|
|
x &= 0x0f;
|
|
|
|
if (x > 9) x += 39;
|
|
|
|
x += 48;
|
|
|
|
|
|
|
|
/* print ASCII char */
|
|
|
|
print_char(x);
|
|
|
|
}
|
2012-05-30 18:13:09 +00:00
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
/**
|
2014-01-20 17:24:57 +00:00
|
|
|
* Print a zero-terminated string s
|
2012-05-30 18:13:09 +00:00
|
|
|
*/
|
2014-01-20 17:24:57 +00:00
|
|
|
Log & operator << (char const * s)
|
2012-05-30 18:13:09 +00:00
|
|
|
{
|
2014-01-20 17:24:57 +00:00
|
|
|
while (*s) print_char(*s++);
|
|
|
|
if (*--s != '\n') { print_char(' '); }
|
2012-05-30 18:13:09 +00:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-01-20 17:24:57 +00:00
|
|
|
* Print an unsigned integer x as hexadecimal value
|
2012-05-30 18:13:09 +00:00
|
|
|
*/
|
2015-02-06 16:23:41 +00:00
|
|
|
Log & operator << (unsigned long const x)
|
2012-05-30 18:13:09 +00:00
|
|
|
{
|
|
|
|
enum {
|
|
|
|
BYTE_WIDTH = 8,
|
|
|
|
CW = sizeof(unsigned char)*BYTE_WIDTH,
|
|
|
|
IW = sizeof(unsigned int)*BYTE_WIDTH
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Walk from most significant to least significant bit and
|
|
|
|
* process 2 hex digits per step.
|
|
|
|
*/
|
|
|
|
bool leading = true;
|
|
|
|
for (int i = IW - CW; i >= 0; i = i - CW)
|
|
|
|
{
|
|
|
|
/* fetch the 2 current digits */
|
|
|
|
unsigned char c = (char)((x >> i) & 0xff);
|
|
|
|
|
|
|
|
/* has the payload part of the value already begun? */
|
|
|
|
if (leading)
|
|
|
|
{
|
|
|
|
/* ignore leading zeros ... */
|
|
|
|
if (!c)
|
|
|
|
{
|
|
|
|
/* ... expect they are the only digits */
|
|
|
|
if (i == 0) _print_4bit_hex(c);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/* transit from leading to payload part */
|
|
|
|
leading = false;
|
|
|
|
if (c < 0x10) {
|
|
|
|
_print_4bit_hex(c);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* print both digits */
|
|
|
|
_print_4bit_hex(c >> 4);
|
|
|
|
_print_4bit_hex(c);
|
|
|
|
}
|
2014-01-20 17:24:57 +00:00
|
|
|
print_char(' ');
|
2012-05-30 18:13:09 +00:00
|
|
|
return *this;
|
|
|
|
}
|
2014-01-20 17:24:57 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Print a pointer p as hexadecimal value
|
|
|
|
*/
|
2015-02-06 16:23:41 +00:00
|
|
|
Log & operator << (void * const p) { return *this << (unsigned long)p; }
|
2012-05-30 18:13:09 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2014-01-20 17:24:57 +00:00
|
|
|
* Return singleton kernel-log
|
2012-05-30 18:13:09 +00:00
|
|
|
*/
|
2014-01-20 17:24:57 +00:00
|
|
|
inline Log & log()
|
2012-05-30 18:13:09 +00:00
|
|
|
{
|
2014-01-20 17:24:57 +00:00
|
|
|
static Log s;
|
|
|
|
return s;
|
2012-05-30 18:13:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-14 16:29:34 +00:00
|
|
|
#endif /* _KERNEL__LOG_H_ */
|
2012-05-30 18:13:09 +00:00
|
|
|
|