2019-06-01 11:08:17 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2010-08-25 22:44:07 +04:00
/*
* linux / drivers / char / ttyprintk . c
*
* Copyright ( C ) 2010 Samo Pogacnik
*/
/*
* This pseudo device allows user to make printk messages . It is possible
* to store " console " messages inline with kernel messages for better analyses
* of the boot process , for example .
*/
# include <linux/device.h>
# include <linux/serial.h>
# include <linux/tty.h>
2014-04-02 16:45:22 +04:00
# include <linux/module.h>
2010-08-25 22:44:07 +04:00
struct ttyprintk_port {
struct tty_port port ;
struct mutex port_write_mutex ;
} ;
static struct ttyprintk_port tpk_port ;
/*
* Our simple preformatting supports transparent output of ( time - stamped )
* printk messages ( also suitable for logging service ) :
* - any cr is replaced by nl
* - adds a ttyprintk source tag in front of each line
2016-09-06 19:49:04 +03:00
* - too long message is fragmented , with ' \ ' nl between fragments
* - TPK_STR_SIZE isn ' t really the write_room limiting factor , because
2010-08-25 22:44:07 +04:00
* it is emptied on the fly during preformatting .
*/
# define TPK_STR_SIZE 508 /* should be bigger then max expected line length */
# define TPK_MAX_ROOM 4096 /* we could assume 4K for instance */
2018-11-07 01:11:37 +03:00
# define TPK_PREFIX KERN_SOH __stringify(CONFIG_TTY_PRINTK_LEVEL)
2010-08-25 22:44:07 +04:00
static int tpk_curr ;
2016-09-06 19:49:04 +03:00
static char tpk_buffer [ TPK_STR_SIZE + 4 ] ;
static void tpk_flush ( void )
{
if ( tpk_curr > 0 ) {
tpk_buffer [ tpk_curr ] = ' \0 ' ;
2018-11-07 01:11:37 +03:00
printk ( TPK_PREFIX " [U] %s \n " , tpk_buffer ) ;
2016-09-06 19:49:04 +03:00
tpk_curr = 0 ;
}
}
2010-08-25 22:44:07 +04:00
static int tpk_printk ( const unsigned char * buf , int count )
{
int i = tpk_curr ;
if ( buf = = NULL ) {
2016-09-06 19:49:04 +03:00
tpk_flush ( ) ;
2010-08-25 22:44:07 +04:00
return i ;
}
for ( i = 0 ; i < count ; i + + ) {
2016-09-06 19:49:04 +03:00
if ( tpk_curr > = TPK_STR_SIZE ) {
2010-08-25 22:44:07 +04:00
/* end of tmp buffer reached: cut the message in two */
2016-09-06 19:49:04 +03:00
tpk_buffer [ tpk_curr + + ] = ' \\ ' ;
tpk_flush ( ) ;
}
switch ( buf [ i ] ) {
case ' \r ' :
tpk_flush ( ) ;
if ( ( i + 1 ) < count & & buf [ i + 1 ] = = ' \n ' )
i + + ;
break ;
case ' \n ' :
tpk_flush ( ) ;
break ;
default :
tpk_buffer [ tpk_curr + + ] = buf [ i ] ;
break ;
2010-08-25 22:44:07 +04:00
}
}
return count ;
}
/*
* TTY operations open function .
*/
static int tpk_open ( struct tty_struct * tty , struct file * filp )
{
tty - > driver_data = & tpk_port ;
return tty_port_open ( & tpk_port . port , tty , filp ) ;
}
/*
* TTY operations close function .
*/
static void tpk_close ( struct tty_struct * tty , struct file * filp )
{
struct ttyprintk_port * tpkp = tty - > driver_data ;
mutex_lock ( & tpkp - > port_write_mutex ) ;
/* flush tpk_printk buffer */
tpk_printk ( NULL , 0 ) ;
mutex_unlock ( & tpkp - > port_write_mutex ) ;
tty_port_close ( & tpkp - > port , tty , filp ) ;
}
/*
* TTY operations write function .
*/
static int tpk_write ( struct tty_struct * tty ,
const unsigned char * buf , int count )
{
struct ttyprintk_port * tpkp = tty - > driver_data ;
int ret ;
/* exclusive use of tpk_printk within this tty */
mutex_lock ( & tpkp - > port_write_mutex ) ;
ret = tpk_printk ( buf , count ) ;
mutex_unlock ( & tpkp - > port_write_mutex ) ;
return ret ;
}
/*
* TTY operations write_room function .
*/
static int tpk_write_room ( struct tty_struct * tty )
{
return TPK_MAX_ROOM ;
}
/*
* TTY operations ioctl function .
*/
2011-02-14 19:27:22 +03:00
static int tpk_ioctl ( struct tty_struct * tty ,
2010-08-25 22:44:07 +04:00
unsigned int cmd , unsigned long arg )
{
struct ttyprintk_port * tpkp = tty - > driver_data ;
if ( ! tpkp )
return - EINVAL ;
switch ( cmd ) {
/* Stop TIOCCONS */
case TIOCCONS :
return - EOPNOTSUPP ;
default :
return - ENOIOCTLCMD ;
}
return 0 ;
}
static const struct tty_operations ttyprintk_ops = {
. open = tpk_open ,
. close = tpk_close ,
. write = tpk_write ,
. write_room = tpk_write_room ,
. ioctl = tpk_ioctl ,
} ;
2015-12-15 02:50:19 +03:00
static const struct tty_port_operations null_ops = { } ;
2010-08-25 22:44:07 +04:00
static struct tty_driver * ttyprintk_driver ;
static int __init ttyprintk_init ( void )
{
int ret = - ENOMEM ;
2012-08-07 23:47:40 +04:00
mutex_init ( & tpk_port . port_write_mutex ) ;
2012-08-09 00:26:43 +04:00
ttyprintk_driver = tty_alloc_driver ( 1 ,
TTY_DRIVER_RESET_TERMIOS |
TTY_DRIVER_REAL_RAW |
TTY_DRIVER_UNNUMBERED_NODE ) ;
2012-08-16 17:16:56 +04:00
if ( IS_ERR ( ttyprintk_driver ) )
return PTR_ERR ( ttyprintk_driver ) ;
2010-08-25 22:44:07 +04:00
2012-11-15 12:49:56 +04:00
tty_port_init ( & tpk_port . port ) ;
2013-05-11 02:40:13 +04:00
tpk_port . port . ops = & null_ops ;
2012-11-15 12:49:56 +04:00
2010-08-25 22:44:07 +04:00
ttyprintk_driver - > driver_name = " ttyprintk " ;
ttyprintk_driver - > name = " ttyprintk " ;
ttyprintk_driver - > major = TTYAUX_MAJOR ;
ttyprintk_driver - > minor_start = 3 ;
ttyprintk_driver - > type = TTY_DRIVER_TYPE_CONSOLE ;
ttyprintk_driver - > init_termios = tty_std_termios ;
ttyprintk_driver - > init_termios . c_oflag = OPOST | OCRNL | ONOCR | ONLRET ;
tty_set_operations ( ttyprintk_driver , & ttyprintk_ops ) ;
2012-08-07 23:47:51 +04:00
tty_port_link_device ( & tpk_port . port , ttyprintk_driver , 0 ) ;
2010-08-25 22:44:07 +04:00
ret = tty_register_driver ( ttyprintk_driver ) ;
if ( ret < 0 ) {
printk ( KERN_ERR " Couldn't register ttyprintk driver \n " ) ;
goto error ;
}
return 0 ;
error :
put_tty_driver ( ttyprintk_driver ) ;
2012-11-15 12:49:56 +04:00
tty_port_destroy ( & tpk_port . port ) ;
2010-08-25 22:44:07 +04:00
return ret ;
}
2014-04-02 16:45:22 +04:00
static void __exit ttyprintk_exit ( void )
{
tty_unregister_driver ( ttyprintk_driver ) ;
put_tty_driver ( ttyprintk_driver ) ;
tty_port_destroy ( & tpk_port . port ) ;
}
2014-01-12 21:37:56 +04:00
device_initcall ( ttyprintk_init ) ;
2014-04-02 16:45:22 +04:00
module_exit ( ttyprintk_exit ) ;
MODULE_LICENSE ( " GPL " ) ;