2005-04-16 15:20:36 -07:00
/*
2009-06-16 10:30:40 +02:00
* driver : reading from and writing to system console on S / 390 via SCLP
2005-04-16 15:20:36 -07:00
*
2009-06-16 10:30:40 +02:00
* Copyright IBM Corp . 1999 , 2009
*
* Author ( s ) : Martin Peschke < mpeschke @ de . ibm . com >
* Martin Schwidefsky < schwidefsky @ de . ibm . com >
2005-04-16 15:20:36 -07:00
*/
# include <linux/kmod.h>
# include <linux/types.h>
# include <linux/err.h>
# include <linux/string.h>
# include <linux/spinlock.h>
# include <linux/ctype.h>
# include <asm/uaccess.h>
# include "sclp.h"
# include "sclp_rw.h"
/*
* The room for the SCCB ( only for writing ) is not equal to a pages size
2007-05-09 08:57:56 +02:00
* ( as it is specified as the maximum size in the SCLP documentation )
2005-04-16 15:20:36 -07:00
* because of the additional data structure described above .
*/
# define MAX_SCCB_ROOM (PAGE_SIZE - sizeof(struct sclp_buffer))
2009-06-16 10:30:40 +02:00
static void sclp_rw_pm_event ( struct sclp_register * reg ,
enum sclp_pm_event sclp_pm_event )
{
sclp_console_pm_event ( sclp_pm_event ) ;
}
2005-04-16 15:20:36 -07:00
/* Event type structure for write message and write priority message */
static struct sclp_register sclp_rw_event = {
2014-09-09 12:53:12 +02:00
. send_mask = EVTYP_MSG_MASK ,
2009-06-16 10:30:40 +02:00
. pm_event_fn = sclp_rw_pm_event ,
2005-04-16 15:20:36 -07:00
} ;
/*
* Setup a sclp write buffer . Gets a page as input ( 4 K ) and returns
* a pointer to a struct sclp_buffer structure that is located at the
* end of the input page . This reduces the buffer space by a few
* bytes but simplifies things .
*/
struct sclp_buffer *
sclp_make_buffer ( void * page , unsigned short columns , unsigned short htab )
{
struct sclp_buffer * buffer ;
2015-10-01 14:11:35 +02:00
struct sccb_header * sccb ;
2005-04-16 15:20:36 -07:00
2015-10-01 14:11:35 +02:00
sccb = ( struct sccb_header * ) page ;
2005-04-16 15:20:36 -07:00
/*
* We keep the struct sclp_buffer structure at the end
* of the sccb page .
*/
buffer = ( ( struct sclp_buffer * ) ( ( addr_t ) sccb + PAGE_SIZE ) ) - 1 ;
buffer - > sccb = sccb ;
buffer - > retry_count = 0 ;
2015-10-01 14:11:35 +02:00
buffer - > messages = 0 ;
buffer - > char_sum = 0 ;
2005-04-16 15:20:36 -07:00
buffer - > current_line = NULL ;
buffer - > current_length = 0 ;
buffer - > columns = columns ;
buffer - > htab = htab ;
/* initialize sccb */
2015-10-01 14:11:35 +02:00
memset ( sccb , 0 , sizeof ( struct sccb_header ) ) ;
sccb - > length = sizeof ( struct sccb_header ) ;
2005-04-16 15:20:36 -07:00
return buffer ;
}
/*
2008-01-26 14:11:20 +01:00
* Return a pointer to the original page that has been used to create
2005-04-16 15:20:36 -07:00
* the buffer .
*/
void *
sclp_unmake_buffer ( struct sclp_buffer * buffer )
{
return buffer - > sccb ;
}
/*
2015-10-01 14:11:35 +02:00
* Initialize a new message the end of the provided buffer with
* enough room for max_len characters . Return 0 on success .
2005-04-16 15:20:36 -07:00
*/
static int
sclp_initialize_mto ( struct sclp_buffer * buffer , int max_len )
{
2015-10-01 14:11:35 +02:00
struct sccb_header * sccb ;
struct msg_buf * msg ;
struct mdb * mdb ;
struct go * go ;
2005-04-16 15:20:36 -07:00
struct mto * mto ;
2015-10-01 14:11:35 +02:00
int msg_size ;
2005-04-16 15:20:36 -07:00
2015-10-01 14:11:35 +02:00
/* max size of new message including message text */
msg_size = sizeof ( struct msg_buf ) + max_len ;
2005-04-16 15:20:36 -07:00
/* check if current buffer sccb can contain the mto */
sccb = buffer - > sccb ;
2015-10-01 14:11:35 +02:00
if ( ( MAX_SCCB_ROOM - sccb - > length ) < msg_size )
2005-04-16 15:20:36 -07:00
return - ENOMEM ;
2015-10-01 14:11:35 +02:00
msg = ( struct msg_buf * ) ( ( addr_t ) sccb + sccb - > length ) ;
memset ( msg , 0 , sizeof ( struct msg_buf ) ) ;
msg - > header . length = sizeof ( struct msg_buf ) ;
msg - > header . type = EVTYP_MSG ;
2005-04-16 15:20:36 -07:00
2015-10-01 14:11:35 +02:00
mdb = & msg - > mdb ;
mdb - > header . length = sizeof ( struct mdb ) ;
mdb - > header . type = 1 ;
mdb - > header . tag = 0xD4C4C240 ; /* ebcdic "MDB " */
mdb - > header . revision_code = 1 ;
go = & mdb - > go ;
go - > length = sizeof ( struct go ) ;
go - > type = 1 ;
mto = & mdb - > mto ;
2005-04-16 15:20:36 -07:00
mto - > length = sizeof ( struct mto ) ;
mto - > type = 4 ; /* message text object */
2007-04-27 16:01:53 +02:00
mto - > line_type_flags = LNTPFLGS_ENDTEXT ; /* end text */
2005-04-16 15:20:36 -07:00
/* set pointer to first byte after struct mto. */
2015-10-01 14:11:35 +02:00
buffer - > current_msg = msg ;
2005-04-16 15:20:36 -07:00
buffer - > current_line = ( char * ) ( mto + 1 ) ;
buffer - > current_length = 0 ;
return 0 ;
}
/*
2015-10-01 14:11:35 +02:00
* Finalize message initialized by sclp_initialize_mto ( ) ,
* updating the sizes of MTO , enclosing MDB , event buffer and SCCB .
2005-04-16 15:20:36 -07:00
*/
static void
sclp_finalize_mto ( struct sclp_buffer * buffer )
{
2015-10-01 14:11:35 +02:00
struct sccb_header * sccb ;
struct msg_buf * msg ;
2005-04-16 15:20:36 -07:00
/*
* update values of sizes
* ( SCCB , Event ( Message ) Buffer , Message Data Block )
*/
2015-10-01 14:11:35 +02:00
sccb = buffer - > sccb ;
msg = buffer - > current_msg ;
msg - > header . length + = buffer - > current_length ;
msg - > mdb . header . length + = buffer - > current_length ;
msg - > mdb . mto . length + = buffer - > current_length ;
sccb - > length + = msg - > header . length ;
2005-04-16 15:20:36 -07:00
/*
* count number of buffered messages ( = number of Message Text
* Objects ) and number of buffered characters
* for the SCCB currently used for buffering and at all
*/
2015-10-01 14:11:35 +02:00
buffer - > messages + + ;
buffer - > char_sum + = buffer - > current_length ;
buffer - > current_line = NULL ;
buffer - > current_length = 0 ;
buffer - > current_msg = NULL ;
2005-04-16 15:20:36 -07:00
}
/*
* processing of a message including escape characters ,
* returns number of characters written to the output sccb
* ( " processed " means that is not guaranteed that the character have already
* been sent to the SCLP but that it will be done at least next time the SCLP
* is not busy )
*/
int
sclp_write ( struct sclp_buffer * buffer , const unsigned char * msg , int count )
{
int spaces , i_msg ;
int rc ;
/*
* parse msg for escape sequences ( \ t , \ v . . . ) and put formated
* msg into an mto ( created by sclp_initialize_mto ) .
*
* We have to do this work ourselfs because there is no support for
* these characters on the native machine and only partial support
* under VM ( Why does VM interpret \ n but the native machine doesn ' t ? )
*
* Depending on i / o - control setting the message is always written
* immediately or we wait for a final new line maybe coming with the
* next message . Besides we avoid a buffer overrun by writing its
* content .
*
* RESTRICTIONS :
*
* \ r and \ b work within one line because we are not able to modify
* previous output that have already been accepted by the SCLP .
*
* \ t combined with following \ r is not correctly represented because
* \ t is expanded to some spaces but \ r does not know about a
* previous \ t and decreases the current position by one column .
* This is in order to a slim and quick implementation .
*/
for ( i_msg = 0 ; i_msg < count ; i_msg + + ) {
switch ( msg [ i_msg ] ) {
case ' \n ' : /* new line, line feed (ASCII) */
/* check if new mto needs to be created */
if ( buffer - > current_line = = NULL ) {
rc = sclp_initialize_mto ( buffer , 0 ) ;
if ( rc )
return i_msg ;
}
sclp_finalize_mto ( buffer ) ;
break ;
case ' \a ' : /* bell, one for several times */
/* set SCLP sound alarm bit in General Object */
2015-10-01 14:11:35 +02:00
if ( buffer - > current_line = = NULL ) {
rc = sclp_initialize_mto ( buffer ,
buffer - > columns ) ;
if ( rc )
return i_msg ;
}
buffer - > current_msg - > mdb . go . general_msg_flags | =
2007-04-27 16:01:53 +02:00
GNRLMSGFLGS_SNDALRM ;
2005-04-16 15:20:36 -07:00
break ;
case ' \t ' : /* horizontal tabulator */
/* check if new mto needs to be created */
if ( buffer - > current_line = = NULL ) {
rc = sclp_initialize_mto ( buffer ,
buffer - > columns ) ;
if ( rc )
return i_msg ;
}
/* "go to (next htab-boundary + 1, same line)" */
do {
if ( buffer - > current_length > = buffer - > columns )
break ;
/* ok, add a blank */
* buffer - > current_line + + = 0x40 ;
buffer - > current_length + + ;
} while ( buffer - > current_length % buffer - > htab ) ;
break ;
case ' \f ' : /* form feed */
case ' \v ' : /* vertical tabulator */
/* "go to (actual column, actual line + 1)" */
/* = new line, leading spaces */
if ( buffer - > current_line ! = NULL ) {
spaces = buffer - > current_length ;
sclp_finalize_mto ( buffer ) ;
rc = sclp_initialize_mto ( buffer ,
buffer - > columns ) ;
if ( rc )
return i_msg ;
memset ( buffer - > current_line , 0x40 , spaces ) ;
buffer - > current_line + = spaces ;
buffer - > current_length = spaces ;
} else {
/* one an empty line this is the same as \n */
rc = sclp_initialize_mto ( buffer ,
buffer - > columns ) ;
if ( rc )
return i_msg ;
sclp_finalize_mto ( buffer ) ;
}
break ;
case ' \b ' : /* backspace */
/* "go to (actual column - 1, actual line)" */
/* decrement counter indicating position, */
/* do not remove last character */
if ( buffer - > current_line ! = NULL & &
buffer - > current_length > 0 ) {
buffer - > current_length - - ;
buffer - > current_line - - ;
}
break ;
case 0x00 : /* end of string */
/* transfer current line to SCCB */
if ( buffer - > current_line ! = NULL )
sclp_finalize_mto ( buffer ) ;
/* skip the rest of the message including the 0 byte */
i_msg = count - 1 ;
break ;
default : /* no escape character */
/* do not output unprintable characters */
if ( ! isprint ( msg [ i_msg ] ) )
break ;
/* check if new mto needs to be created */
if ( buffer - > current_line = = NULL ) {
rc = sclp_initialize_mto ( buffer ,
buffer - > columns ) ;
if ( rc )
return i_msg ;
}
* buffer - > current_line + + = sclp_ascebc ( msg [ i_msg ] ) ;
buffer - > current_length + + ;
break ;
}
/* check if current mto is full */
if ( buffer - > current_line ! = NULL & &
buffer - > current_length > = buffer - > columns )
sclp_finalize_mto ( buffer ) ;
}
/* return number of processed characters */
return i_msg ;
}
/*
* Return the number of free bytes in the sccb
*/
int
sclp_buffer_space ( struct sclp_buffer * buffer )
{
2015-10-01 14:11:35 +02:00
struct sccb_header * sccb ;
2005-04-16 15:20:36 -07:00
int count ;
2015-10-01 14:11:35 +02:00
sccb = buffer - > sccb ;
count = MAX_SCCB_ROOM - sccb - > length ;
2005-04-16 15:20:36 -07:00
if ( buffer - > current_line ! = NULL )
2015-10-01 14:11:35 +02:00
count - = sizeof ( struct msg_buf ) + buffer - > current_length ;
2005-04-16 15:20:36 -07:00
return count ;
}
/*
* Return number of characters in buffer
*/
int
sclp_chars_in_buffer ( struct sclp_buffer * buffer )
{
int count ;
2015-10-01 14:11:35 +02:00
count = buffer - > char_sum ;
2005-04-16 15:20:36 -07:00
if ( buffer - > current_line ! = NULL )
count + = buffer - > current_length ;
return count ;
}
/*
* sets or provides some values that influence the drivers behaviour
*/
void
sclp_set_columns ( struct sclp_buffer * buffer , unsigned short columns )
{
buffer - > columns = columns ;
if ( buffer - > current_line ! = NULL & &
buffer - > current_length > buffer - > columns )
sclp_finalize_mto ( buffer ) ;
}
void
sclp_set_htab ( struct sclp_buffer * buffer , unsigned short htab )
{
buffer - > htab = htab ;
}
/*
* called by sclp_console_init and / or sclp_tty_init
*/
int
sclp_rw_init ( void )
{
static int init_done = 0 ;
int rc ;
if ( init_done )
return 0 ;
rc = sclp_register ( & sclp_rw_event ) ;
if ( rc = = 0 )
init_done = 1 ;
return rc ;
}
# define SCLP_BUFFER_MAX_RETRY 1
/*
* second half of Write Event Data - function that has to be done after
* interruption indicating completion of Service Call .
*/
static void
sclp_writedata_callback ( struct sclp_req * request , void * data )
{
int rc ;
struct sclp_buffer * buffer ;
2015-10-01 14:11:35 +02:00
struct sccb_header * sccb ;
2005-04-16 15:20:36 -07:00
buffer = ( struct sclp_buffer * ) data ;
sccb = buffer - > sccb ;
if ( request - > status = = SCLP_REQ_FAILED ) {
if ( buffer - > callback ! = NULL )
buffer - > callback ( buffer , - EIO ) ;
return ;
}
/* check SCLP response code and choose suitable action */
2015-10-01 14:11:35 +02:00
switch ( sccb - > response_code ) {
2005-04-16 15:20:36 -07:00
case 0x0020 :
/* Normal completion, buffer processed, message(s) sent */
rc = 0 ;
break ;
case 0x0340 : /* Contained SCLP equipment check */
if ( + + buffer - > retry_count > SCLP_BUFFER_MAX_RETRY ) {
rc = - EIO ;
break ;
}
/* remove processed buffers and requeue rest */
if ( sclp_remove_processed ( ( struct sccb_header * ) sccb ) > 0 ) {
/* not all buffers were processed */
2015-10-01 14:11:35 +02:00
sccb - > response_code = 0x0000 ;
2005-04-16 15:20:36 -07:00
buffer - > request . status = SCLP_REQ_FILLED ;
rc = sclp_add_request ( request ) ;
if ( rc = = 0 )
return ;
} else
rc = 0 ;
break ;
case 0x0040 : /* SCLP equipment check */
case 0x05f0 : /* Target resource in improper state */
if ( + + buffer - > retry_count > SCLP_BUFFER_MAX_RETRY ) {
rc = - EIO ;
break ;
}
/* retry request */
2015-10-01 14:11:35 +02:00
sccb - > response_code = 0x0000 ;
2005-04-16 15:20:36 -07:00
buffer - > request . status = SCLP_REQ_FILLED ;
rc = sclp_add_request ( request ) ;
if ( rc = = 0 )
return ;
break ;
default :
2015-10-01 14:11:35 +02:00
if ( sccb - > response_code = = 0x71f0 )
2005-04-16 15:20:36 -07:00
rc = - ENOMEM ;
else
rc = - EINVAL ;
break ;
}
if ( buffer - > callback ! = NULL )
buffer - > callback ( buffer , rc ) ;
}
/*
* Setup the request structure in the struct sclp_buffer to do SCLP Write
* Event Data and pass the request to the core SCLP loop . Return zero on
* success , non - zero otherwise .
*/
int
sclp_emit_buffer ( struct sclp_buffer * buffer ,
void ( * callback ) ( struct sclp_buffer * , int ) )
{
/* add current line if there is one */
if ( buffer - > current_line ! = NULL )
sclp_finalize_mto ( buffer ) ;
/* Are there messages in the output buffer ? */
2015-10-01 14:11:35 +02:00
if ( buffer - > messages = = 0 )
2005-04-16 15:20:36 -07:00
return - EIO ;
2007-02-05 21:18:37 +01:00
buffer - > request . command = SCLP_CMDW_WRITE_EVENT_DATA ;
2005-04-16 15:20:36 -07:00
buffer - > request . status = SCLP_REQ_FILLED ;
buffer - > request . callback = sclp_writedata_callback ;
buffer - > request . callback_data = buffer ;
2015-10-01 14:11:35 +02:00
buffer - > request . sccb = buffer - > sccb ;
2005-04-16 15:20:36 -07:00
buffer - > callback = callback ;
return sclp_add_request ( & buffer - > request ) ;
}