2005-04-17 02:20:36 +04:00
/*
* procfs handler for Linux I2O subsystem
*
* ( c ) Copyright 1999 Deepak Saxena
*
* Originally written by Deepak Saxena ( deepak @ plexity . net )
*
* 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 .
*
* This is an initial test release . The code is based on the design of the
* ide procfs system ( drivers / block / ide - proc . c ) . Some code taken from
* i2o - core module by Alan Cox .
*
* DISCLAIMER : This code is still under development / test and may cause
* your system to behave unpredictably . Use at your own discretion .
*
*
* Fixes / additions :
* Juha Siev <EFBFBD> nen ( Juha . Sievanen @ cs . Helsinki . FI ) ,
* Auvo H <EFBFBD> kkinen ( Auvo . Hakkinen @ cs . Helsinki . FI )
* University of Helsinki , Department of Computer Science
* LAN entries
* Markus Lidel < Markus . Lidel @ shadowconnect . com >
* Changes for new I2O API
*/
# define OSM_NAME "proc-osm"
2006-01-06 11:19:32 +03:00
# define OSM_VERSION "1.316"
2005-04-17 02:20:36 +04:00
# define OSM_DESCRIPTION "I2O ProcFS OSM"
# define I2O_MAX_MODULES 4
// FIXME!
# define FMT_U64_HEX "0x%08x%08x"
# define U64_VAL(pu64) *((u32*)(pu64)+1), *((u32*)(pu64))
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/pci.h>
# include <linux/i2o.h>
# include <linux/proc_fs.h>
# include <linux/seq_file.h>
# include <linux/init.h>
# include <linux/module.h>
# include <linux/errno.h>
# include <linux/spinlock.h>
# include <linux/workqueue.h>
# include <asm/io.h>
# include <asm/uaccess.h>
# include <asm/byteorder.h>
/* Structure used to define /proc entries */
typedef struct _i2o_proc_entry_t {
char * name ; /* entry name */
mode_t mode ; /* mode */
2006-03-28 13:56:41 +04:00
const struct file_operations * fops ; /* open function */
2005-04-17 02:20:36 +04:00
} i2o_proc_entry ;
/* global I2O /proc/i2o entry */
static struct proc_dir_entry * i2o_proc_dir_root ;
/* proc OSM driver struct */
static struct i2o_driver i2o_proc_driver = {
. name = OSM_NAME ,
} ;
static int print_serial_number ( struct seq_file * seq , u8 * serialno , int max_len )
{
int i ;
/* 19990419 -sralston
* The I2O v1 .5 ( and v2 .0 so far ) " official specification "
* got serial numbers WRONG !
* Apparently , and despite what Section 3.4 .4 says and
* Figure 3 - 35 shows ( pg 3 - 39 in the pdf doc ) ,
* the convention / consensus seems to be :
* + First byte is SNFormat
* + Second byte is SNLen ( but only if SNFormat = = 7 ( ? ) )
* + ( v2 .0 ) SCSI + BS may use IEEE Registered ( 64 or 128 bit ) format
*/
switch ( serialno [ 0 ] ) {
case I2O_SNFORMAT_BINARY : /* Binary */
seq_printf ( seq , " 0x " ) ;
for ( i = 0 ; i < serialno [ 1 ] ; i + + ) {
seq_printf ( seq , " %02X " , serialno [ 2 + i ] ) ;
}
break ;
case I2O_SNFORMAT_ASCII : /* ASCII */
if ( serialno [ 1 ] < ' ' ) { /* printable or SNLen? */
/* sanity */
max_len =
( max_len < serialno [ 1 ] ) ? max_len : serialno [ 1 ] ;
serialno [ 1 + max_len ] = ' \0 ' ;
/* just print it */
seq_printf ( seq , " %s " , & serialno [ 2 ] ) ;
} else {
/* print chars for specified length */
for ( i = 0 ; i < serialno [ 1 ] ; i + + ) {
seq_printf ( seq , " %c " , serialno [ 2 + i ] ) ;
}
}
break ;
case I2O_SNFORMAT_UNICODE : /* UNICODE */
seq_printf ( seq , " UNICODE Format. Can't Display \n " ) ;
break ;
case I2O_SNFORMAT_LAN48_MAC : /* LAN-48 MAC Address */
seq_printf ( seq ,
" LAN-48 MAC address @ %02X:%02X:%02X:%02X:%02X:%02X " ,
serialno [ 2 ] , serialno [ 3 ] ,
serialno [ 4 ] , serialno [ 5 ] , serialno [ 6 ] , serialno [ 7 ] ) ;
break ;
case I2O_SNFORMAT_WAN : /* WAN MAC Address */
/* FIXME: Figure out what a WAN access address looks like?? */
seq_printf ( seq , " WAN Access Address " ) ;
break ;
/* plus new in v2.0 */
case I2O_SNFORMAT_LAN64_MAC : /* LAN-64 MAC Address */
/* FIXME: Figure out what a LAN-64 address really looks like?? */
seq_printf ( seq ,
" LAN-64 MAC address @ [?:%02X:%02X:?] %02X:%02X:%02X:%02X:%02X:%02X " ,
serialno [ 8 ] , serialno [ 9 ] ,
serialno [ 2 ] , serialno [ 3 ] ,
serialno [ 4 ] , serialno [ 5 ] , serialno [ 6 ] , serialno [ 7 ] ) ;
break ;
case I2O_SNFORMAT_DDM : /* I2O DDM */
seq_printf ( seq ,
" DDM: Tid=%03Xh, Rsvd=%04Xh, OrgId=%04Xh " ,
* ( u16 * ) & serialno [ 2 ] ,
* ( u16 * ) & serialno [ 4 ] , * ( u16 * ) & serialno [ 6 ] ) ;
break ;
case I2O_SNFORMAT_IEEE_REG64 : /* IEEE Registered (64-bit) */
case I2O_SNFORMAT_IEEE_REG128 : /* IEEE Registered (128-bit) */
/* FIXME: Figure if this is even close?? */
seq_printf ( seq ,
" IEEE NodeName(hi,lo)=(%08Xh:%08Xh), PortName(hi,lo)=(%08Xh:%08Xh) \n " ,
* ( u32 * ) & serialno [ 2 ] ,
* ( u32 * ) & serialno [ 6 ] ,
* ( u32 * ) & serialno [ 10 ] , * ( u32 * ) & serialno [ 14 ] ) ;
break ;
case I2O_SNFORMAT_UNKNOWN : /* Unknown 0 */
case I2O_SNFORMAT_UNKNOWN2 : /* Unknown 0xff */
default :
seq_printf ( seq , " Unknown data format (0x%02x) " , serialno [ 0 ] ) ;
break ;
}
return 0 ;
}
/**
* i2o_get_class_name - do i2o class name lookup
* @ class : class number
*
2006-12-07 07:38:43 +03:00
* Return a descriptive string for an i2o class .
2005-04-17 02:20:36 +04:00
*/
static const char * i2o_get_class_name ( int class )
{
int idx = 16 ;
static char * i2o_class_name [ ] = {
" Executive " ,
" Device Driver Module " ,
" Block Device " ,
" Tape Device " ,
" LAN Interface " ,
" WAN Interface " ,
" Fibre Channel Port " ,
" Fibre Channel Device " ,
" SCSI Device " ,
" ATE Port " ,
" ATE Device " ,
" Floppy Controller " ,
" Floppy Device " ,
" Secondary Bus Port " ,
" Peer Transport Agent " ,
" Peer Transport " ,
" Unknown "
} ;
switch ( class & 0xfff ) {
case I2O_CLASS_EXECUTIVE :
idx = 0 ;
break ;
case I2O_CLASS_DDM :
idx = 1 ;
break ;
case I2O_CLASS_RANDOM_BLOCK_STORAGE :
idx = 2 ;
break ;
case I2O_CLASS_SEQUENTIAL_STORAGE :
idx = 3 ;
break ;
case I2O_CLASS_LAN :
idx = 4 ;
break ;
case I2O_CLASS_WAN :
idx = 5 ;
break ;
case I2O_CLASS_FIBRE_CHANNEL_PORT :
idx = 6 ;
break ;
case I2O_CLASS_FIBRE_CHANNEL_PERIPHERAL :
idx = 7 ;
break ;
case I2O_CLASS_SCSI_PERIPHERAL :
idx = 8 ;
break ;
case I2O_CLASS_ATE_PORT :
idx = 9 ;
break ;
case I2O_CLASS_ATE_PERIPHERAL :
idx = 10 ;
break ;
case I2O_CLASS_FLOPPY_CONTROLLER :
idx = 11 ;
break ;
case I2O_CLASS_FLOPPY_DEVICE :
idx = 12 ;
break ;
2005-06-24 09:02:16 +04:00
case I2O_CLASS_BUS_ADAPTER :
2005-04-17 02:20:36 +04:00
idx = 13 ;
break ;
case I2O_CLASS_PEER_TRANSPORT_AGENT :
idx = 14 ;
break ;
case I2O_CLASS_PEER_TRANSPORT :
idx = 15 ;
break ;
}
return i2o_class_name [ idx ] ;
}
# define SCSI_TABLE_SIZE 13
static char * scsi_devices [ ] = {
" Direct-Access Read/Write " ,
" Sequential-Access Storage " ,
" Printer " ,
" Processor " ,
" WORM Device " ,
" CD-ROM Device " ,
" Scanner Device " ,
" Optical Memory Device " ,
" Medium Changer Device " ,
" Communications Device " ,
" Graphics Art Pre-Press Device " ,
" Graphics Art Pre-Press Device " ,
" Array Controller Device "
} ;
static char * chtostr ( u8 * chars , int n )
{
char tmp [ 256 ] ;
tmp [ 0 ] = 0 ;
return strncat ( tmp , ( char * ) chars , n ) ;
}
static int i2o_report_query_status ( struct seq_file * seq , int block_status ,
char * group )
{
switch ( block_status ) {
case - ETIMEDOUT :
return seq_printf ( seq , " Timeout reading group %s. \n " , group ) ;
case - ENOMEM :
return seq_printf ( seq , " No free memory to read the table. \n " ) ;
case - I2O_PARAMS_STATUS_INVALID_GROUP_ID :
return seq_printf ( seq , " Group %s not supported. \n " , group ) ;
default :
return seq_printf ( seq ,
" Error reading group %s. BlockStatus 0x%02X \n " ,
group , - block_status ) ;
}
}
static char * bus_strings [ ] = {
" Local Bus " ,
" ISA " ,
" EISA " ,
" MCA " ,
" PCI " ,
" PCMCIA " ,
" NUBUS " ,
" CARDBUS "
} ;
static int i2o_seq_show_hrt ( struct seq_file * seq , void * v )
{
struct i2o_controller * c = ( struct i2o_controller * ) seq - > private ;
i2o_hrt * hrt = ( i2o_hrt * ) c - > hrt . virt ;
u32 bus ;
int i ;
if ( hrt - > hrt_version ) {
seq_printf ( seq ,
" HRT table for controller is too new a version. \n " ) ;
return 0 ;
}
seq_printf ( seq , " HRT has %d entries of %d bytes each. \n " ,
hrt - > num_entries , hrt - > entry_len < < 2 ) ;
for ( i = 0 ; i < hrt - > num_entries ; i + + ) {
seq_printf ( seq , " Entry %d: \n " , i ) ;
seq_printf ( seq , " Adapter ID: %0#10x \n " ,
hrt - > hrt_entry [ i ] . adapter_id ) ;
seq_printf ( seq , " Controlling tid: %0#6x \n " ,
hrt - > hrt_entry [ i ] . parent_tid ) ;
if ( hrt - > hrt_entry [ i ] . bus_type ! = 0x80 ) {
bus = hrt - > hrt_entry [ i ] . bus_type ;
seq_printf ( seq , " %s Information \n " ,
bus_strings [ bus ] ) ;
switch ( bus ) {
case I2O_BUS_LOCAL :
seq_printf ( seq , " IOBase: %0#6x, " ,
hrt - > hrt_entry [ i ] . bus . local_bus .
LbBaseIOPort ) ;
seq_printf ( seq , " MemoryBase: %0#10x \n " ,
hrt - > hrt_entry [ i ] . bus . local_bus .
LbBaseMemoryAddress ) ;
break ;
case I2O_BUS_ISA :
seq_printf ( seq , " IOBase: %0#6x, " ,
hrt - > hrt_entry [ i ] . bus . isa_bus .
IsaBaseIOPort ) ;
seq_printf ( seq , " MemoryBase: %0#10x, " ,
hrt - > hrt_entry [ i ] . bus . isa_bus .
IsaBaseMemoryAddress ) ;
seq_printf ( seq , " CSN: %0#4x, " ,
hrt - > hrt_entry [ i ] . bus . isa_bus . CSN ) ;
break ;
case I2O_BUS_EISA :
seq_printf ( seq , " IOBase: %0#6x, " ,
hrt - > hrt_entry [ i ] . bus . eisa_bus .
EisaBaseIOPort ) ;
seq_printf ( seq , " MemoryBase: %0#10x, " ,
hrt - > hrt_entry [ i ] . bus . eisa_bus .
EisaBaseMemoryAddress ) ;
seq_printf ( seq , " Slot: %0#4x, " ,
hrt - > hrt_entry [ i ] . bus . eisa_bus .
EisaSlotNumber ) ;
break ;
case I2O_BUS_MCA :
seq_printf ( seq , " IOBase: %0#6x, " ,
hrt - > hrt_entry [ i ] . bus . mca_bus .
McaBaseIOPort ) ;
seq_printf ( seq , " MemoryBase: %0#10x, " ,
hrt - > hrt_entry [ i ] . bus . mca_bus .
McaBaseMemoryAddress ) ;
seq_printf ( seq , " Slot: %0#4x, " ,
hrt - > hrt_entry [ i ] . bus . mca_bus .
McaSlotNumber ) ;
break ;
case I2O_BUS_PCI :
seq_printf ( seq , " Bus: %0#4x " ,
hrt - > hrt_entry [ i ] . bus . pci_bus .
PciBusNumber ) ;
seq_printf ( seq , " Dev: %0#4x " ,
hrt - > hrt_entry [ i ] . bus . pci_bus .
PciDeviceNumber ) ;
seq_printf ( seq , " Func: %0#4x " ,
hrt - > hrt_entry [ i ] . bus . pci_bus .
PciFunctionNumber ) ;
seq_printf ( seq , " Vendor: %0#6x " ,
hrt - > hrt_entry [ i ] . bus . pci_bus .
PciVendorID ) ;
seq_printf ( seq , " Device: %0#6x \n " ,
hrt - > hrt_entry [ i ] . bus . pci_bus .
PciDeviceID ) ;
break ;
default :
seq_printf ( seq , " Unsupported Bus Type \n " ) ;
}
} else
seq_printf ( seq , " Unknown Bus Type \n " ) ;
}
return 0 ;
}
static int i2o_seq_show_lct ( struct seq_file * seq , void * v )
{
struct i2o_controller * c = ( struct i2o_controller * ) seq - > private ;
i2o_lct * lct = ( i2o_lct * ) c - > lct ;
int entries ;
int i ;
# define BUS_TABLE_SIZE 3
static char * bus_ports [ ] = {
" Generic Bus " ,
" SCSI Bus " ,
" Fibre Channel Bus "
} ;
entries = ( lct - > table_size - 3 ) / 9 ;
seq_printf ( seq , " LCT contains %d %s \n " , entries ,
entries = = 1 ? " entry " : " entries " ) ;
if ( lct - > boot_tid )
seq_printf ( seq , " Boot Device @ ID %d \n " , lct - > boot_tid ) ;
seq_printf ( seq , " Current Change Indicator: %#10x \n " , lct - > change_ind ) ;
for ( i = 0 ; i < entries ; i + + ) {
seq_printf ( seq , " Entry %d \n " , i ) ;
seq_printf ( seq , " Class, SubClass : %s " ,
i2o_get_class_name ( lct - > lct_entry [ i ] . class_id ) ) ;
/*
* Classes which we ' ll print subclass info for
*/
switch ( lct - > lct_entry [ i ] . class_id & 0xFFF ) {
case I2O_CLASS_RANDOM_BLOCK_STORAGE :
switch ( lct - > lct_entry [ i ] . sub_class ) {
case 0x00 :
seq_printf ( seq , " , Direct-Access Read/Write " ) ;
break ;
case 0x04 :
seq_printf ( seq , " , WORM Drive " ) ;
break ;
case 0x05 :
seq_printf ( seq , " , CD-ROM Drive " ) ;
break ;
case 0x07 :
seq_printf ( seq , " , Optical Memory Device " ) ;
break ;
default :
seq_printf ( seq , " , Unknown (0x%02x) " ,
lct - > lct_entry [ i ] . sub_class ) ;
break ;
}
break ;
case I2O_CLASS_LAN :
switch ( lct - > lct_entry [ i ] . sub_class & 0xFF ) {
case 0x30 :
seq_printf ( seq , " , Ethernet " ) ;
break ;
case 0x40 :
seq_printf ( seq , " , 100base VG " ) ;
break ;
case 0x50 :
seq_printf ( seq , " , IEEE 802.5/Token-Ring " ) ;
break ;
case 0x60 :
seq_printf ( seq , " , ANSI X3T9.5 FDDI " ) ;
break ;
case 0x70 :
seq_printf ( seq , " , Fibre Channel " ) ;
break ;
default :
seq_printf ( seq , " , Unknown Sub-Class (0x%02x) " ,
lct - > lct_entry [ i ] . sub_class & 0xFF ) ;
break ;
}
break ;
case I2O_CLASS_SCSI_PERIPHERAL :
if ( lct - > lct_entry [ i ] . sub_class < SCSI_TABLE_SIZE )
seq_printf ( seq , " , %s " ,
scsi_devices [ lct - > lct_entry [ i ] .
sub_class ] ) ;
else
seq_printf ( seq , " , Unknown Device Type " ) ;
break ;
2005-06-24 09:02:16 +04:00
case I2O_CLASS_BUS_ADAPTER :
2005-04-17 02:20:36 +04:00
if ( lct - > lct_entry [ i ] . sub_class < BUS_TABLE_SIZE )
seq_printf ( seq , " , %s " ,
bus_ports [ lct - > lct_entry [ i ] .
sub_class ] ) ;
else
seq_printf ( seq , " , Unknown Bus Type " ) ;
break ;
}
seq_printf ( seq , " \n " ) ;
seq_printf ( seq , " Local TID : 0x%03x \n " ,
lct - > lct_entry [ i ] . tid ) ;
seq_printf ( seq , " User TID : 0x%03x \n " ,
lct - > lct_entry [ i ] . user_tid ) ;
seq_printf ( seq , " Parent TID : 0x%03x \n " ,
lct - > lct_entry [ i ] . parent_tid ) ;
seq_printf ( seq , " Identity Tag : 0x%x%x%x%x%x%x%x%x \n " ,
lct - > lct_entry [ i ] . identity_tag [ 0 ] ,
lct - > lct_entry [ i ] . identity_tag [ 1 ] ,
lct - > lct_entry [ i ] . identity_tag [ 2 ] ,
lct - > lct_entry [ i ] . identity_tag [ 3 ] ,
lct - > lct_entry [ i ] . identity_tag [ 4 ] ,
lct - > lct_entry [ i ] . identity_tag [ 5 ] ,
lct - > lct_entry [ i ] . identity_tag [ 6 ] ,
lct - > lct_entry [ i ] . identity_tag [ 7 ] ) ;
seq_printf ( seq , " Change Indicator : %0#10x \n " ,
lct - > lct_entry [ i ] . change_ind ) ;
seq_printf ( seq , " Event Capab Mask : %0#10x \n " ,
lct - > lct_entry [ i ] . device_flags ) ;
}
return 0 ;
}
static int i2o_seq_show_status ( struct seq_file * seq , void * v )
{
struct i2o_controller * c = ( struct i2o_controller * ) seq - > private ;
char prodstr [ 25 ] ;
int version ;
i2o_status_block * sb = c - > status_block . virt ;
i2o_status_get ( c ) ; // reread the status block
seq_printf ( seq , " Organization ID : %0#6x \n " , sb - > org_id ) ;
version = sb - > i2o_version ;
/* FIXME for Spec 2.0
if ( version = = 0x02 ) {
seq_printf ( seq , " Lowest I2O version supported: " ) ;
switch ( workspace [ 2 ] ) {
case 0x00 :
seq_printf ( seq , " 1.0 \n " ) ;
break ;
case 0x01 :
seq_printf ( seq , " 1.5 \n " ) ;
break ;
case 0x02 :
seq_printf ( seq , " 2.0 \n " ) ;
break ;
}
seq_printf ( seq , " Highest I2O version supported: " ) ;
switch ( workspace [ 3 ] ) {
case 0x00 :
seq_printf ( seq , " 1.0 \n " ) ;
break ;
case 0x01 :
seq_printf ( seq , " 1.5 \n " ) ;
break ;
case 0x02 :
seq_printf ( seq , " 2.0 \n " ) ;
break ;
}
}
*/
seq_printf ( seq , " IOP ID : %0#5x \n " , sb - > iop_id ) ;
seq_printf ( seq , " Host Unit ID : %0#6x \n " , sb - > host_unit_id ) ;
seq_printf ( seq , " Segment Number : %0#5x \n " , sb - > segment_number ) ;
seq_printf ( seq , " I2O version : " ) ;
switch ( version ) {
case 0x00 :
seq_printf ( seq , " 1.0 \n " ) ;
break ;
case 0x01 :
seq_printf ( seq , " 1.5 \n " ) ;
break ;
case 0x02 :
seq_printf ( seq , " 2.0 \n " ) ;
break ;
default :
seq_printf ( seq , " Unknown version \n " ) ;
}
seq_printf ( seq , " IOP State : " ) ;
switch ( sb - > iop_state ) {
case 0x01 :
seq_printf ( seq , " INIT \n " ) ;
break ;
case 0x02 :
seq_printf ( seq , " RESET \n " ) ;
break ;
case 0x04 :
seq_printf ( seq , " HOLD \n " ) ;
break ;
case 0x05 :
seq_printf ( seq , " READY \n " ) ;
break ;
case 0x08 :
seq_printf ( seq , " OPERATIONAL \n " ) ;
break ;
case 0x10 :
seq_printf ( seq , " FAILED \n " ) ;
break ;
case 0x11 :
seq_printf ( seq , " FAULTED \n " ) ;
break ;
default :
seq_printf ( seq , " Unknown \n " ) ;
break ;
}
seq_printf ( seq , " Messenger Type : " ) ;
switch ( sb - > msg_type ) {
case 0x00 :
seq_printf ( seq , " Memory mapped \n " ) ;
break ;
case 0x01 :
seq_printf ( seq , " Memory mapped only \n " ) ;
break ;
case 0x02 :
seq_printf ( seq , " Remote only \n " ) ;
break ;
case 0x03 :
seq_printf ( seq , " Memory mapped and remote \n " ) ;
break ;
default :
seq_printf ( seq , " Unknown \n " ) ;
}
seq_printf ( seq , " Inbound Frame Size : %d bytes \n " ,
sb - > inbound_frame_size < < 2 ) ;
seq_printf ( seq , " Max Inbound Frames : %d \n " ,
sb - > max_inbound_frames ) ;
seq_printf ( seq , " Current Inbound Frames : %d \n " ,
sb - > cur_inbound_frames ) ;
seq_printf ( seq , " Max Outbound Frames : %d \n " ,
sb - > max_outbound_frames ) ;
/* Spec doesn't say if NULL terminated or not... */
memcpy ( prodstr , sb - > product_id , 24 ) ;
prodstr [ 24 ] = ' \0 ' ;
seq_printf ( seq , " Product ID : %s \n " , prodstr ) ;
seq_printf ( seq , " Expected LCT Size : %d bytes \n " ,
sb - > expected_lct_size ) ;
seq_printf ( seq , " IOP Capabilities \n " ) ;
seq_printf ( seq , " Context Field Size Support : " ) ;
switch ( sb - > iop_capabilities & 0x0000003 ) {
case 0 :
seq_printf ( seq , " Supports only 32-bit context fields \n " ) ;
break ;
case 1 :
seq_printf ( seq , " Supports only 64-bit context fields \n " ) ;
break ;
case 2 :
seq_printf ( seq , " Supports 32-bit and 64-bit context fields, "
" but not concurrently \n " ) ;
break ;
case 3 :
seq_printf ( seq , " Supports 32-bit and 64-bit context fields "
" concurrently \n " ) ;
break ;
default :
seq_printf ( seq , " 0x%08x \n " , sb - > iop_capabilities ) ;
}
seq_printf ( seq , " Current Context Field Size : " ) ;
switch ( sb - > iop_capabilities & 0x0000000C ) {
case 0 :
seq_printf ( seq , " not configured \n " ) ;
break ;
case 4 :
seq_printf ( seq , " Supports only 32-bit context fields \n " ) ;
break ;
case 8 :
seq_printf ( seq , " Supports only 64-bit context fields \n " ) ;
break ;
case 12 :
seq_printf ( seq , " Supports both 32-bit or 64-bit context fields "
" concurrently \n " ) ;
break ;
default :
seq_printf ( seq , " \n " ) ;
}
seq_printf ( seq , " Inbound Peer Support : %s \n " ,
( sb - >
iop_capabilities & 0x00000010 ) ? " Supported " :
" Not supported " ) ;
seq_printf ( seq , " Outbound Peer Support : %s \n " ,
( sb - >
iop_capabilities & 0x00000020 ) ? " Supported " :
" Not supported " ) ;
seq_printf ( seq , " Peer to Peer Support : %s \n " ,
( sb - >
iop_capabilities & 0x00000040 ) ? " Supported " :
" Not supported " ) ;
seq_printf ( seq , " Desired private memory size : %d kB \n " ,
sb - > desired_mem_size > > 10 ) ;
seq_printf ( seq , " Allocated private memory size : %d kB \n " ,
sb - > current_mem_size > > 10 ) ;
seq_printf ( seq , " Private memory base address : %0#10x \n " ,
sb - > current_mem_base ) ;
seq_printf ( seq , " Desired private I/O size : %d kB \n " ,
sb - > desired_io_size > > 10 ) ;
seq_printf ( seq , " Allocated private I/O size : %d kB \n " ,
sb - > current_io_size > > 10 ) ;
seq_printf ( seq , " Private I/O base address : %0#10x \n " ,
sb - > current_io_base ) ;
return 0 ;
}
static int i2o_seq_show_hw ( struct seq_file * seq , void * v )
{
struct i2o_controller * c = ( struct i2o_controller * ) seq - > private ;
static u32 work32 [ 5 ] ;
static u8 * work8 = ( u8 * ) work32 ;
static u16 * work16 = ( u16 * ) work32 ;
int token ;
u32 hwcap ;
static char * cpu_table [ ] = {
" Intel 80960 series " ,
" AMD2900 series " ,
" Motorola 68000 series " ,
" ARM series " ,
" MIPS series " ,
" Sparc series " ,
" PowerPC series " ,
" Intel x86 series "
} ;
token =
i2o_parm_field_get ( c - > exec , 0x0000 , - 1 , & work32 , sizeof ( work32 ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token , " 0x0000 IOP Hardware " ) ;
return 0 ;
}
seq_printf ( seq , " I2O Vendor ID : %0#6x \n " , work16 [ 0 ] ) ;
seq_printf ( seq , " Product ID : %0#6x \n " , work16 [ 1 ] ) ;
seq_printf ( seq , " CPU : " ) ;
if ( work8 [ 16 ] > 8 )
seq_printf ( seq , " Unknown \n " ) ;
else
seq_printf ( seq , " %s \n " , cpu_table [ work8 [ 16 ] ] ) ;
/* Anyone using ProcessorVersion? */
seq_printf ( seq , " RAM : %dkB \n " , work32 [ 1 ] > > 10 ) ;
seq_printf ( seq , " Non-Volatile Mem : %dkB \n " , work32 [ 2 ] > > 10 ) ;
hwcap = work32 [ 3 ] ;
seq_printf ( seq , " Capabilities : 0x%08x \n " , hwcap ) ;
seq_printf ( seq , " [%s] Self booting \n " ,
( hwcap & 0x00000001 ) ? " + " : " - " ) ;
seq_printf ( seq , " [%s] Upgradable IRTOS \n " ,
( hwcap & 0x00000002 ) ? " + " : " - " ) ;
seq_printf ( seq , " [%s] Supports downloading DDMs \n " ,
( hwcap & 0x00000004 ) ? " + " : " - " ) ;
seq_printf ( seq , " [%s] Supports installing DDMs \n " ,
( hwcap & 0x00000008 ) ? " + " : " - " ) ;
seq_printf ( seq , " [%s] Battery-backed RAM \n " ,
( hwcap & 0x00000010 ) ? " + " : " - " ) ;
return 0 ;
}
/* Executive group 0003h - Executing DDM List (table) */
static int i2o_seq_show_ddm_table ( struct seq_file * seq , void * v )
{
struct i2o_controller * c = ( struct i2o_controller * ) seq - > private ;
int token ;
int i ;
typedef struct _i2o_exec_execute_ddm_table {
u16 ddm_tid ;
u8 module_type ;
u8 reserved ;
u16 i2o_vendor_id ;
u16 module_id ;
u8 module_name_version [ 28 ] ;
u32 data_size ;
u32 code_size ;
} i2o_exec_execute_ddm_table ;
struct {
u16 result_count ;
u16 pad ;
u16 block_size ;
u8 block_status ;
u8 error_info_size ;
u16 row_count ;
u16 more_flag ;
i2o_exec_execute_ddm_table ddm_table [ I2O_MAX_MODULES ] ;
} * result ;
i2o_exec_execute_ddm_table ddm_table ;
result = kmalloc ( sizeof ( * result ) , GFP_KERNEL ) ;
if ( ! result )
return - ENOMEM ;
token = i2o_parm_table_get ( c - > exec , I2O_PARAMS_TABLE_GET , 0x0003 , - 1 ,
NULL , 0 , result , sizeof ( * result ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token ,
" 0x0003 Executing DDM List " ) ;
goto out ;
}
seq_printf ( seq ,
" Tid Module_type Vendor Mod_id Module_name Vrs Data_size Code_size \n " ) ;
ddm_table = result - > ddm_table [ 0 ] ;
for ( i = 0 ; i < result - > row_count ; ddm_table = result - > ddm_table [ + + i ] ) {
seq_printf ( seq , " 0x%03x " , ddm_table . ddm_tid & 0xFFF ) ;
switch ( ddm_table . module_type ) {
case 0x01 :
seq_printf ( seq , " Downloaded DDM " ) ;
break ;
case 0x22 :
seq_printf ( seq , " Embedded DDM " ) ;
break ;
default :
seq_printf ( seq , " " ) ;
}
seq_printf ( seq , " %-#7x " , ddm_table . i2o_vendor_id ) ;
seq_printf ( seq , " %-#8x " , ddm_table . module_id ) ;
seq_printf ( seq , " %-29s " ,
chtostr ( ddm_table . module_name_version , 28 ) ) ;
seq_printf ( seq , " %9d " , ddm_table . data_size ) ;
seq_printf ( seq , " %8d " , ddm_table . code_size ) ;
seq_printf ( seq , " \n " ) ;
}
out :
kfree ( result ) ;
return 0 ;
}
/* Executive group 0004h - Driver Store (scalar) */
static int i2o_seq_show_driver_store ( struct seq_file * seq , void * v )
{
struct i2o_controller * c = ( struct i2o_controller * ) seq - > private ;
u32 work32 [ 8 ] ;
int token ;
token =
i2o_parm_field_get ( c - > exec , 0x0004 , - 1 , & work32 , sizeof ( work32 ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token , " 0x0004 Driver Store " ) ;
return 0 ;
}
seq_printf ( seq , " Module limit : %d \n "
" Module count : %d \n "
" Current space : %d kB \n "
" Free space : %d kB \n " ,
work32 [ 0 ] , work32 [ 1 ] , work32 [ 2 ] > > 10 , work32 [ 3 ] > > 10 ) ;
return 0 ;
}
/* Executive group 0005h - Driver Store Table (table) */
static int i2o_seq_show_drivers_stored ( struct seq_file * seq , void * v )
{
typedef struct _i2o_driver_store {
u16 stored_ddm_index ;
u8 module_type ;
u8 reserved ;
u16 i2o_vendor_id ;
u16 module_id ;
u8 module_name_version [ 28 ] ;
u8 date [ 8 ] ;
u32 module_size ;
u32 mpb_size ;
u32 module_flags ;
} i2o_driver_store_table ;
struct i2o_controller * c = ( struct i2o_controller * ) seq - > private ;
int token ;
int i ;
typedef struct {
u16 result_count ;
u16 pad ;
u16 block_size ;
u8 block_status ;
u8 error_info_size ;
u16 row_count ;
u16 more_flag ;
i2o_driver_store_table dst [ I2O_MAX_MODULES ] ;
} i2o_driver_result_table ;
i2o_driver_result_table * result ;
i2o_driver_store_table * dst ;
result = kmalloc ( sizeof ( i2o_driver_result_table ) , GFP_KERNEL ) ;
if ( result = = NULL )
return - ENOMEM ;
token = i2o_parm_table_get ( c - > exec , I2O_PARAMS_TABLE_GET , 0x0005 , - 1 ,
NULL , 0 , result , sizeof ( * result ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token ,
" 0x0005 DRIVER STORE TABLE " ) ;
kfree ( result ) ;
return 0 ;
}
seq_printf ( seq ,
" # Module_type Vendor Mod_id Module_name Vrs "
" Date Mod_size Par_size Flags \n " ) ;
for ( i = 0 , dst = & result - > dst [ 0 ] ; i < result - > row_count ;
dst = & result - > dst [ + + i ] ) {
seq_printf ( seq , " %-3d " , dst - > stored_ddm_index ) ;
switch ( dst - > module_type ) {
case 0x01 :
seq_printf ( seq , " Downloaded DDM " ) ;
break ;
case 0x22 :
seq_printf ( seq , " Embedded DDM " ) ;
break ;
default :
seq_printf ( seq , " " ) ;
}
seq_printf ( seq , " %-#7x " , dst - > i2o_vendor_id ) ;
seq_printf ( seq , " %-#8x " , dst - > module_id ) ;
seq_printf ( seq , " %-29s " , chtostr ( dst - > module_name_version , 28 ) ) ;
seq_printf ( seq , " %-9s " , chtostr ( dst - > date , 8 ) ) ;
seq_printf ( seq , " %8d " , dst - > module_size ) ;
seq_printf ( seq , " %8d " , dst - > mpb_size ) ;
seq_printf ( seq , " 0x%04x " , dst - > module_flags ) ;
seq_printf ( seq , " \n " ) ;
}
kfree ( result ) ;
return 0 ;
}
/* Generic group F000h - Params Descriptor (table) */
static int i2o_seq_show_groups ( struct seq_file * seq , void * v )
{
struct i2o_device * d = ( struct i2o_device * ) seq - > private ;
int token ;
int i ;
u8 properties ;
typedef struct _i2o_group_info {
u16 group_number ;
u16 field_count ;
u16 row_count ;
u8 properties ;
u8 reserved ;
} i2o_group_info ;
struct {
u16 result_count ;
u16 pad ;
u16 block_size ;
u8 block_status ;
u8 error_info_size ;
u16 row_count ;
u16 more_flag ;
i2o_group_info group [ 256 ] ;
} * result ;
result = kmalloc ( sizeof ( * result ) , GFP_KERNEL ) ;
if ( ! result )
return - ENOMEM ;
token = i2o_parm_table_get ( d , I2O_PARAMS_TABLE_GET , 0xF000 , - 1 , NULL , 0 ,
result , sizeof ( * result ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token , " 0xF000 Params Descriptor " ) ;
goto out ;
}
seq_printf ( seq ,
" # Group FieldCount RowCount Type Add Del Clear \n " ) ;
for ( i = 0 ; i < result - > row_count ; i + + ) {
seq_printf ( seq , " %-3d " , i ) ;
seq_printf ( seq , " 0x%04X " , result - > group [ i ] . group_number ) ;
seq_printf ( seq , " %10d " , result - > group [ i ] . field_count ) ;
seq_printf ( seq , " %8d " , result - > group [ i ] . row_count ) ;
properties = result - > group [ i ] . properties ;
if ( properties & 0x1 )
seq_printf ( seq , " Table " ) ;
else
seq_printf ( seq , " Scalar " ) ;
if ( properties & 0x2 )
seq_printf ( seq , " + " ) ;
else
seq_printf ( seq , " - " ) ;
if ( properties & 0x4 )
seq_printf ( seq , " + " ) ;
else
seq_printf ( seq , " - " ) ;
if ( properties & 0x8 )
seq_printf ( seq , " + " ) ;
else
seq_printf ( seq , " - " ) ;
seq_printf ( seq , " \n " ) ;
}
if ( result - > more_flag )
seq_printf ( seq , " There is more... \n " ) ;
out :
kfree ( result ) ;
return 0 ;
}
/* Generic group F001h - Physical Device Table (table) */
static int i2o_seq_show_phys_device ( struct seq_file * seq , void * v )
{
struct i2o_device * d = ( struct i2o_device * ) seq - > private ;
int token ;
int i ;
struct {
u16 result_count ;
u16 pad ;
u16 block_size ;
u8 block_status ;
u8 error_info_size ;
u16 row_count ;
u16 more_flag ;
u32 adapter_id [ 64 ] ;
} result ;
token = i2o_parm_table_get ( d , I2O_PARAMS_TABLE_GET , 0xF001 , - 1 , NULL , 0 ,
& result , sizeof ( result ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token ,
" 0xF001 Physical Device Table " ) ;
return 0 ;
}
if ( result . row_count )
seq_printf ( seq , " # AdapterId \n " ) ;
for ( i = 0 ; i < result . row_count ; i + + ) {
seq_printf ( seq , " %-2d " , i ) ;
seq_printf ( seq , " %#7x \n " , result . adapter_id [ i ] ) ;
}
if ( result . more_flag )
seq_printf ( seq , " There is more... \n " ) ;
return 0 ;
}
/* Generic group F002h - Claimed Table (table) */
static int i2o_seq_show_claimed ( struct seq_file * seq , void * v )
{
struct i2o_device * d = ( struct i2o_device * ) seq - > private ;
int token ;
int i ;
struct {
u16 result_count ;
u16 pad ;
u16 block_size ;
u8 block_status ;
u8 error_info_size ;
u16 row_count ;
u16 more_flag ;
u16 claimed_tid [ 64 ] ;
} result ;
token = i2o_parm_table_get ( d , I2O_PARAMS_TABLE_GET , 0xF002 , - 1 , NULL , 0 ,
& result , sizeof ( result ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token , " 0xF002 Claimed Table " ) ;
return 0 ;
}
if ( result . row_count )
seq_printf ( seq , " # ClaimedTid \n " ) ;
for ( i = 0 ; i < result . row_count ; i + + ) {
seq_printf ( seq , " %-2d " , i ) ;
seq_printf ( seq , " %#7x \n " , result . claimed_tid [ i ] ) ;
}
if ( result . more_flag )
seq_printf ( seq , " There is more... \n " ) ;
return 0 ;
}
/* Generic group F003h - User Table (table) */
static int i2o_seq_show_users ( struct seq_file * seq , void * v )
{
struct i2o_device * d = ( struct i2o_device * ) seq - > private ;
int token ;
int i ;
typedef struct _i2o_user_table {
u16 instance ;
u16 user_tid ;
u8 claim_type ;
u8 reserved1 ;
u16 reserved2 ;
} i2o_user_table ;
struct {
u16 result_count ;
u16 pad ;
u16 block_size ;
u8 block_status ;
u8 error_info_size ;
u16 row_count ;
u16 more_flag ;
i2o_user_table user [ 64 ] ;
} * result ;
result = kmalloc ( sizeof ( * result ) , GFP_KERNEL ) ;
if ( ! result )
return - ENOMEM ;
token = i2o_parm_table_get ( d , I2O_PARAMS_TABLE_GET , 0xF003 , - 1 , NULL , 0 ,
result , sizeof ( * result ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token , " 0xF003 User Table " ) ;
goto out ;
}
seq_printf ( seq , " # Instance UserTid ClaimType \n " ) ;
for ( i = 0 ; i < result - > row_count ; i + + ) {
seq_printf ( seq , " %-3d " , i ) ;
seq_printf ( seq , " %#8x " , result - > user [ i ] . instance ) ;
seq_printf ( seq , " %#7x " , result - > user [ i ] . user_tid ) ;
seq_printf ( seq , " %#9x \n " , result - > user [ i ] . claim_type ) ;
}
if ( result - > more_flag )
seq_printf ( seq , " There is more... \n " ) ;
out :
kfree ( result ) ;
return 0 ;
}
/* Generic group F005h - Private message extensions (table) (optional) */
static int i2o_seq_show_priv_msgs ( struct seq_file * seq , void * v )
{
struct i2o_device * d = ( struct i2o_device * ) seq - > private ;
int token ;
int i ;
typedef struct _i2o_private {
u16 ext_instance ;
u16 organization_id ;
u16 x_function_code ;
} i2o_private ;
struct {
u16 result_count ;
u16 pad ;
u16 block_size ;
u8 block_status ;
u8 error_info_size ;
u16 row_count ;
u16 more_flag ;
i2o_private extension [ 64 ] ;
} result ;
token = i2o_parm_table_get ( d , I2O_PARAMS_TABLE_GET , 0xF000 , - 1 , NULL , 0 ,
& result , sizeof ( result ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token ,
" 0xF005 Private Message Extensions (optional) " ) ;
return 0 ;
}
seq_printf ( seq , " Instance# OrgId FunctionCode \n " ) ;
for ( i = 0 ; i < result . row_count ; i + + ) {
seq_printf ( seq , " %0#9x " , result . extension [ i ] . ext_instance ) ;
seq_printf ( seq , " %0#6x " , result . extension [ i ] . organization_id ) ;
seq_printf ( seq , " %0#6x " , result . extension [ i ] . x_function_code ) ;
seq_printf ( seq , " \n " ) ;
}
if ( result . more_flag )
seq_printf ( seq , " There is more... \n " ) ;
return 0 ;
}
/* Generic group F006h - Authorized User Table (table) */
static int i2o_seq_show_authorized_users ( struct seq_file * seq , void * v )
{
struct i2o_device * d = ( struct i2o_device * ) seq - > private ;
int token ;
int i ;
struct {
u16 result_count ;
u16 pad ;
u16 block_size ;
u8 block_status ;
u8 error_info_size ;
u16 row_count ;
u16 more_flag ;
u32 alternate_tid [ 64 ] ;
} result ;
token = i2o_parm_table_get ( d , I2O_PARAMS_TABLE_GET , 0xF006 , - 1 , NULL , 0 ,
& result , sizeof ( result ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token ,
" 0xF006 Autohorized User Table " ) ;
return 0 ;
}
if ( result . row_count )
seq_printf ( seq , " # AlternateTid \n " ) ;
for ( i = 0 ; i < result . row_count ; i + + ) {
seq_printf ( seq , " %-2d " , i ) ;
seq_printf ( seq , " %#7x " , result . alternate_tid [ i ] ) ;
}
if ( result . more_flag )
seq_printf ( seq , " There is more... \n " ) ;
return 0 ;
}
/* Generic group F100h - Device Identity (scalar) */
static int i2o_seq_show_dev_identity ( struct seq_file * seq , void * v )
{
struct i2o_device * d = ( struct i2o_device * ) seq - > private ;
static u32 work32 [ 128 ] ; // allow for "stuff" + up to 256 byte (max) serial number
// == (allow) 512d bytes (max)
static u16 * work16 = ( u16 * ) work32 ;
int token ;
token = i2o_parm_field_get ( d , 0xF100 , - 1 , & work32 , sizeof ( work32 ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token , " 0xF100 Device Identity " ) ;
return 0 ;
}
seq_printf ( seq , " Device Class : %s \n " , i2o_get_class_name ( work16 [ 0 ] ) ) ;
seq_printf ( seq , " Owner TID : %0#5x \n " , work16 [ 2 ] ) ;
seq_printf ( seq , " Parent TID : %0#5x \n " , work16 [ 3 ] ) ;
seq_printf ( seq , " Vendor info : %s \n " ,
chtostr ( ( u8 * ) ( work32 + 2 ) , 16 ) ) ;
seq_printf ( seq , " Product info : %s \n " ,
chtostr ( ( u8 * ) ( work32 + 6 ) , 16 ) ) ;
seq_printf ( seq , " Description : %s \n " ,
chtostr ( ( u8 * ) ( work32 + 10 ) , 16 ) ) ;
seq_printf ( seq , " Product rev. : %s \n " ,
chtostr ( ( u8 * ) ( work32 + 14 ) , 8 ) ) ;
seq_printf ( seq , " Serial number : " ) ;
print_serial_number ( seq , ( u8 * ) ( work32 + 16 ) ,
/* allow for SNLen plus
* possible trailing ' \0 '
*/
sizeof ( work32 ) - ( 16 * sizeof ( u32 ) ) - 2 ) ;
seq_printf ( seq , " \n " ) ;
return 0 ;
}
static int i2o_seq_show_dev_name ( struct seq_file * seq , void * v )
{
struct i2o_device * d = ( struct i2o_device * ) seq - > private ;
seq_printf ( seq , " %s \n " , d - > device . bus_id ) ;
return 0 ;
}
/* Generic group F101h - DDM Identity (scalar) */
static int i2o_seq_show_ddm_identity ( struct seq_file * seq , void * v )
{
struct i2o_device * d = ( struct i2o_device * ) seq - > private ;
int token ;
struct {
u16 ddm_tid ;
u8 module_name [ 24 ] ;
u8 module_rev [ 8 ] ;
u8 sn_format ;
u8 serial_number [ 12 ] ;
u8 pad [ 256 ] ; // allow up to 256 byte (max) serial number
} result ;
token = i2o_parm_field_get ( d , 0xF101 , - 1 , & result , sizeof ( result ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token , " 0xF101 DDM Identity " ) ;
return 0 ;
}
seq_printf ( seq , " Registering DDM TID : 0x%03x \n " , result . ddm_tid ) ;
seq_printf ( seq , " Module name : %s \n " ,
chtostr ( result . module_name , 24 ) ) ;
seq_printf ( seq , " Module revision : %s \n " ,
chtostr ( result . module_rev , 8 ) ) ;
seq_printf ( seq , " Serial number : " ) ;
print_serial_number ( seq , result . serial_number , sizeof ( result ) - 36 ) ;
/* allow for SNLen plus possible trailing '\0' */
seq_printf ( seq , " \n " ) ;
return 0 ;
}
/* Generic group F102h - User Information (scalar) */
static int i2o_seq_show_uinfo ( struct seq_file * seq , void * v )
{
struct i2o_device * d = ( struct i2o_device * ) seq - > private ;
int token ;
struct {
u8 device_name [ 64 ] ;
u8 service_name [ 64 ] ;
u8 physical_location [ 64 ] ;
u8 instance_number [ 4 ] ;
} result ;
token = i2o_parm_field_get ( d , 0xF102 , - 1 , & result , sizeof ( result ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token , " 0xF102 User Information " ) ;
return 0 ;
}
seq_printf ( seq , " Device name : %s \n " ,
chtostr ( result . device_name , 64 ) ) ;
seq_printf ( seq , " Service name : %s \n " ,
chtostr ( result . service_name , 64 ) ) ;
seq_printf ( seq , " Physical name : %s \n " ,
chtostr ( result . physical_location , 64 ) ) ;
seq_printf ( seq , " Instance number : %s \n " ,
chtostr ( result . instance_number , 4 ) ) ;
return 0 ;
}
/* Generic group F103h - SGL Operating Limits (scalar) */
static int i2o_seq_show_sgl_limits ( struct seq_file * seq , void * v )
{
struct i2o_device * d = ( struct i2o_device * ) seq - > private ;
static u32 work32 [ 12 ] ;
static u16 * work16 = ( u16 * ) work32 ;
static u8 * work8 = ( u8 * ) work32 ;
int token ;
token = i2o_parm_field_get ( d , 0xF103 , - 1 , & work32 , sizeof ( work32 ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token ,
" 0xF103 SGL Operating Limits " ) ;
return 0 ;
}
seq_printf ( seq , " SGL chain size : %d \n " , work32 [ 0 ] ) ;
seq_printf ( seq , " Max SGL chain size : %d \n " , work32 [ 1 ] ) ;
seq_printf ( seq , " SGL chain size target : %d \n " , work32 [ 2 ] ) ;
seq_printf ( seq , " SGL frag count : %d \n " , work16 [ 6 ] ) ;
seq_printf ( seq , " Max SGL frag count : %d \n " , work16 [ 7 ] ) ;
seq_printf ( seq , " SGL frag count target : %d \n " , work16 [ 8 ] ) ;
/* FIXME
if ( d - > i2oversion = = 0x02 )
{
*/
seq_printf ( seq , " SGL data alignment : %d \n " , work16 [ 8 ] ) ;
seq_printf ( seq , " SGL addr limit : %d \n " , work8 [ 20 ] ) ;
seq_printf ( seq , " SGL addr sizes supported : " ) ;
if ( work8 [ 21 ] & 0x01 )
seq_printf ( seq , " 32 bit " ) ;
if ( work8 [ 21 ] & 0x02 )
seq_printf ( seq , " 64 bit " ) ;
if ( work8 [ 21 ] & 0x04 )
seq_printf ( seq , " 96 bit " ) ;
if ( work8 [ 21 ] & 0x08 )
seq_printf ( seq , " 128 bit " ) ;
seq_printf ( seq , " \n " ) ;
/*
}
*/
return 0 ;
}
/* Generic group F200h - Sensors (scalar) */
static int i2o_seq_show_sensors ( struct seq_file * seq , void * v )
{
struct i2o_device * d = ( struct i2o_device * ) seq - > private ;
int token ;
struct {
u16 sensor_instance ;
u8 component ;
u16 component_instance ;
u8 sensor_class ;
u8 sensor_type ;
u8 scaling_exponent ;
u32 actual_reading ;
u32 minimum_reading ;
u32 low2lowcat_treshold ;
u32 lowcat2low_treshold ;
u32 lowwarn2low_treshold ;
u32 low2lowwarn_treshold ;
u32 norm2lowwarn_treshold ;
u32 lowwarn2norm_treshold ;
u32 nominal_reading ;
u32 hiwarn2norm_treshold ;
u32 norm2hiwarn_treshold ;
u32 high2hiwarn_treshold ;
u32 hiwarn2high_treshold ;
u32 hicat2high_treshold ;
u32 hi2hicat_treshold ;
u32 maximum_reading ;
u8 sensor_state ;
u16 event_enable ;
} result ;
token = i2o_parm_field_get ( d , 0xF200 , - 1 , & result , sizeof ( result ) ) ;
if ( token < 0 ) {
i2o_report_query_status ( seq , token ,
" 0xF200 Sensors (optional) " ) ;
return 0 ;
}
seq_printf ( seq , " Sensor instance : %d \n " , result . sensor_instance ) ;
seq_printf ( seq , " Component : %d = " , result . component ) ;
switch ( result . component ) {
case 0 :
seq_printf ( seq , " Other " ) ;
break ;
case 1 :
seq_printf ( seq , " Planar logic Board " ) ;
break ;
case 2 :
seq_printf ( seq , " CPU " ) ;
break ;
case 3 :
seq_printf ( seq , " Chassis " ) ;
break ;
case 4 :
seq_printf ( seq , " Power Supply " ) ;
break ;
case 5 :
seq_printf ( seq , " Storage " ) ;
break ;
case 6 :
seq_printf ( seq , " External " ) ;
break ;
}
seq_printf ( seq , " \n " ) ;
seq_printf ( seq , " Component instance : %d \n " ,
result . component_instance ) ;
seq_printf ( seq , " Sensor class : %s \n " ,
result . sensor_class ? " Analog " : " Digital " ) ;
seq_printf ( seq , " Sensor type : %d = " , result . sensor_type ) ;
switch ( result . sensor_type ) {
case 0 :
seq_printf ( seq , " Other \n " ) ;
break ;
case 1 :
seq_printf ( seq , " Thermal \n " ) ;
break ;
case 2 :
seq_printf ( seq , " DC voltage (DC volts) \n " ) ;
break ;
case 3 :
seq_printf ( seq , " AC voltage (AC volts) \n " ) ;
break ;
case 4 :
seq_printf ( seq , " DC current (DC amps) \n " ) ;
break ;
case 5 :
seq_printf ( seq , " AC current (AC volts) \n " ) ;
break ;
case 6 :
seq_printf ( seq , " Door open \n " ) ;
break ;
case 7 :
seq_printf ( seq , " Fan operational \n " ) ;
break ;
}
seq_printf ( seq , " Scaling exponent : %d \n " ,
result . scaling_exponent ) ;
seq_printf ( seq , " Actual reading : %d \n " , result . actual_reading ) ;
seq_printf ( seq , " Minimum reading : %d \n " , result . minimum_reading ) ;
seq_printf ( seq , " Low2LowCat treshold : %d \n " ,
result . low2lowcat_treshold ) ;
seq_printf ( seq , " LowCat2Low treshold : %d \n " ,
result . lowcat2low_treshold ) ;
seq_printf ( seq , " LowWarn2Low treshold : %d \n " ,
result . lowwarn2low_treshold ) ;
seq_printf ( seq , " Low2LowWarn treshold : %d \n " ,
result . low2lowwarn_treshold ) ;
seq_printf ( seq , " Norm2LowWarn treshold : %d \n " ,
result . norm2lowwarn_treshold ) ;
seq_printf ( seq , " LowWarn2Norm treshold : %d \n " ,
result . lowwarn2norm_treshold ) ;
seq_printf ( seq , " Nominal reading : %d \n " , result . nominal_reading ) ;
seq_printf ( seq , " HiWarn2Norm treshold : %d \n " ,
result . hiwarn2norm_treshold ) ;
seq_printf ( seq , " Norm2HiWarn treshold : %d \n " ,
result . norm2hiwarn_treshold ) ;
seq_printf ( seq , " High2HiWarn treshold : %d \n " ,
result . high2hiwarn_treshold ) ;
seq_printf ( seq , " HiWarn2High treshold : %d \n " ,
result . hiwarn2high_treshold ) ;
seq_printf ( seq , " HiCat2High treshold : %d \n " ,
result . hicat2high_treshold ) ;
seq_printf ( seq , " High2HiCat treshold : %d \n " ,
result . hi2hicat_treshold ) ;
seq_printf ( seq , " Maximum reading : %d \n " , result . maximum_reading ) ;
seq_printf ( seq , " Sensor state : %d = " , result . sensor_state ) ;
switch ( result . sensor_state ) {
case 0 :
seq_printf ( seq , " Normal \n " ) ;
break ;
case 1 :
seq_printf ( seq , " Abnormal \n " ) ;
break ;
case 2 :
seq_printf ( seq , " Unknown \n " ) ;
break ;
case 3 :
seq_printf ( seq , " Low Catastrophic (LoCat) \n " ) ;
break ;
case 4 :
seq_printf ( seq , " Low (Low) \n " ) ;
break ;
case 5 :
seq_printf ( seq , " Low Warning (LoWarn) \n " ) ;
break ;
case 6 :
seq_printf ( seq , " High Warning (HiWarn) \n " ) ;
break ;
case 7 :
seq_printf ( seq , " High (High) \n " ) ;
break ;
case 8 :
seq_printf ( seq , " High Catastrophic (HiCat) \n " ) ;
break ;
}
seq_printf ( seq , " Event_enable : 0x%02X \n " , result . event_enable ) ;
seq_printf ( seq , " [%s] Operational state change. \n " ,
( result . event_enable & 0x01 ) ? " + " : " - " ) ;
seq_printf ( seq , " [%s] Low catastrophic. \n " ,
( result . event_enable & 0x02 ) ? " + " : " - " ) ;
seq_printf ( seq , " [%s] Low reading. \n " ,
( result . event_enable & 0x04 ) ? " + " : " - " ) ;
seq_printf ( seq , " [%s] Low warning. \n " ,
( result . event_enable & 0x08 ) ? " + " : " - " ) ;
seq_printf ( seq ,
" [%s] Change back to normal from out of range state. \n " ,
( result . event_enable & 0x10 ) ? " + " : " - " ) ;
seq_printf ( seq , " [%s] High warning. \n " ,
( result . event_enable & 0x20 ) ? " + " : " - " ) ;
seq_printf ( seq , " [%s] High reading. \n " ,
( result . event_enable & 0x40 ) ? " + " : " - " ) ;
seq_printf ( seq , " [%s] High catastrophic. \n " ,
( result . event_enable & 0x80 ) ? " + " : " - " ) ;
return 0 ;
}
static int i2o_seq_open_hrt ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_hrt , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_lct ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_lct , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_status ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_status , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_hw ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_hw , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_ddm_table ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_ddm_table , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_driver_store ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_driver_store , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_drivers_stored ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_drivers_stored , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_groups ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_groups , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_phys_device ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_phys_device , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_claimed ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_claimed , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_users ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_users , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_priv_msgs ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_priv_msgs , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_authorized_users ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_authorized_users ,
PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_dev_identity ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_dev_identity , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_ddm_identity ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_ddm_identity , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_uinfo ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_uinfo , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_sgl_limits ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_sgl_limits , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_sensors ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_sensors , PDE ( inode ) - > data ) ;
} ;
static int i2o_seq_open_dev_name ( struct inode * inode , struct file * file )
{
return single_open ( file , i2o_seq_show_dev_name , PDE ( inode ) - > data ) ;
} ;
static struct file_operations i2o_seq_fops_lct = {
. open = i2o_seq_open_lct ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_hrt = {
. open = i2o_seq_open_hrt ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_status = {
. open = i2o_seq_open_status ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_hw = {
. open = i2o_seq_open_hw ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_ddm_table = {
. open = i2o_seq_open_ddm_table ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_driver_store = {
. open = i2o_seq_open_driver_store ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_drivers_stored = {
. open = i2o_seq_open_drivers_stored ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_groups = {
. open = i2o_seq_open_groups ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_phys_device = {
. open = i2o_seq_open_phys_device ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_claimed = {
. open = i2o_seq_open_claimed ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_users = {
. open = i2o_seq_open_users ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_priv_msgs = {
. open = i2o_seq_open_priv_msgs ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_authorized_users = {
. open = i2o_seq_open_authorized_users ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_dev_name = {
. open = i2o_seq_open_dev_name ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_dev_identity = {
. open = i2o_seq_open_dev_identity ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_ddm_identity = {
. open = i2o_seq_open_ddm_identity ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_uinfo = {
. open = i2o_seq_open_uinfo ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_sgl_limits = {
. open = i2o_seq_open_sgl_limits ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct file_operations i2o_seq_fops_sensors = {
. open = i2o_seq_open_sensors ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
/*
* IOP specific entries . . . write field just in case someone
* ever wants one .
*/
static i2o_proc_entry i2o_proc_generic_iop_entries [ ] = {
{ " hrt " , S_IFREG | S_IRUGO , & i2o_seq_fops_hrt } ,
{ " lct " , S_IFREG | S_IRUGO , & i2o_seq_fops_lct } ,
{ " status " , S_IFREG | S_IRUGO , & i2o_seq_fops_status } ,
{ " hw " , S_IFREG | S_IRUGO , & i2o_seq_fops_hw } ,
{ " ddm_table " , S_IFREG | S_IRUGO , & i2o_seq_fops_ddm_table } ,
{ " driver_store " , S_IFREG | S_IRUGO , & i2o_seq_fops_driver_store } ,
{ " drivers_stored " , S_IFREG | S_IRUGO , & i2o_seq_fops_drivers_stored } ,
{ NULL , 0 , NULL }
} ;
/*
* Device specific entries
*/
static i2o_proc_entry generic_dev_entries [ ] = {
{ " groups " , S_IFREG | S_IRUGO , & i2o_seq_fops_groups } ,
{ " phys_dev " , S_IFREG | S_IRUGO , & i2o_seq_fops_phys_device } ,
{ " claimed " , S_IFREG | S_IRUGO , & i2o_seq_fops_claimed } ,
{ " users " , S_IFREG | S_IRUGO , & i2o_seq_fops_users } ,
{ " priv_msgs " , S_IFREG | S_IRUGO , & i2o_seq_fops_priv_msgs } ,
{ " authorized_users " , S_IFREG | S_IRUGO , & i2o_seq_fops_authorized_users } ,
{ " dev_identity " , S_IFREG | S_IRUGO , & i2o_seq_fops_dev_identity } ,
{ " ddm_identity " , S_IFREG | S_IRUGO , & i2o_seq_fops_ddm_identity } ,
{ " user_info " , S_IFREG | S_IRUGO , & i2o_seq_fops_uinfo } ,
{ " sgl_limits " , S_IFREG | S_IRUGO , & i2o_seq_fops_sgl_limits } ,
{ " sensors " , S_IFREG | S_IRUGO , & i2o_seq_fops_sensors } ,
{ NULL , 0 , NULL }
} ;
/*
* Storage unit specific entries ( SCSI Periph , BS ) with device names
*/
static i2o_proc_entry rbs_dev_entries [ ] = {
{ " dev_name " , S_IFREG | S_IRUGO , & i2o_seq_fops_dev_name } ,
{ NULL , 0 , NULL }
} ;
/**
* i2o_proc_create_entries - Creates proc dir entries
* @ dir : proc dir entry under which the entries should be placed
* @ i2o_pe : pointer to the entries which should be added
* @ data : pointer to I2O controller or device
*
* Create proc dir entries for a I2O controller or I2O device .
*
* Returns 0 on success or negative error code on failure .
*/
static int i2o_proc_create_entries ( struct proc_dir_entry * dir ,
i2o_proc_entry * i2o_pe , void * data )
{
struct proc_dir_entry * tmp ;
while ( i2o_pe - > name ) {
tmp = create_proc_entry ( i2o_pe - > name , i2o_pe - > mode , dir ) ;
if ( ! tmp )
return - 1 ;
tmp - > data = data ;
tmp - > proc_fops = i2o_pe - > fops ;
i2o_pe + + ;
}
return 0 ;
}
/**
* i2o_proc_subdir_remove - Remove child entries from a proc entry
* @ dir : proc dir entry from which the childs should be removed
*
* Iterate over each i2o proc entry under dir and remove it . If the child
* also has entries , remove them too .
*/
static void i2o_proc_subdir_remove ( struct proc_dir_entry * dir )
{
struct proc_dir_entry * pe , * tmp ;
pe = dir - > subdir ;
while ( pe ) {
tmp = pe - > next ;
i2o_proc_subdir_remove ( pe ) ;
remove_proc_entry ( pe - > name , dir ) ;
pe = tmp ;
}
} ;
/**
* i2o_proc_device_add - Add an I2O device to the proc dir
* @ dir : proc dir entry to which the device should be added
* @ dev : I2O device which should be added
*
* Add an I2O device to the proc dir entry dir and create the entries for
* the device depending on the class of the I2O device .
*/
static void i2o_proc_device_add ( struct proc_dir_entry * dir ,
struct i2o_device * dev )
{
char buff [ 10 ] ;
struct proc_dir_entry * devdir ;
i2o_proc_entry * i2o_pe = NULL ;
sprintf ( buff , " %03x " , dev - > lct_data . tid ) ;
osm_debug ( " adding device /proc/i2o/%s/%s \n " , dev - > iop - > name , buff ) ;
devdir = proc_mkdir ( buff , dir ) ;
if ( ! devdir ) {
osm_warn ( " Could not allocate procdir! \n " ) ;
return ;
}
devdir - > data = dev ;
i2o_proc_create_entries ( devdir , generic_dev_entries , dev ) ;
/* Inform core that we want updates about this device's status */
switch ( dev - > lct_data . class_id ) {
case I2O_CLASS_SCSI_PERIPHERAL :
case I2O_CLASS_RANDOM_BLOCK_STORAGE :
i2o_pe = rbs_dev_entries ;
break ;
default :
break ;
}
if ( i2o_pe )
i2o_proc_create_entries ( devdir , i2o_pe , dev ) ;
}
/**
* i2o_proc_iop_add - Add an I2O controller to the i2o proc tree
* @ dir : parent proc dir entry
* @ c : I2O controller which should be added
*
* Add the entries to the parent proc dir entry . Also each device is added
* to the controllers proc dir entry .
*
* Returns 0 on success or negative error code on failure .
*/
static int i2o_proc_iop_add ( struct proc_dir_entry * dir ,
struct i2o_controller * c )
{
struct proc_dir_entry * iopdir ;
struct i2o_device * dev ;
osm_debug ( " adding IOP /proc/i2o/%s \n " , c - > name ) ;
iopdir = proc_mkdir ( c - > name , dir ) ;
if ( ! iopdir )
return - 1 ;
iopdir - > data = c ;
i2o_proc_create_entries ( iopdir , i2o_proc_generic_iop_entries , c ) ;
list_for_each_entry ( dev , & c - > devices , list )
i2o_proc_device_add ( iopdir , dev ) ;
return 0 ;
}
/**
* i2o_proc_iop_remove - Removes an I2O controller from the i2o proc tree
* @ dir : parent proc dir entry
* @ c : I2O controller which should be removed
*
* Iterate over each i2o proc entry and search controller c . If it is found
* remove it from the tree .
*/
static void i2o_proc_iop_remove ( struct proc_dir_entry * dir ,
struct i2o_controller * c )
{
struct proc_dir_entry * pe , * tmp ;
pe = dir - > subdir ;
while ( pe ) {
tmp = pe - > next ;
if ( pe - > data = = c ) {
i2o_proc_subdir_remove ( pe ) ;
remove_proc_entry ( pe - > name , dir ) ;
}
osm_debug ( " removing IOP /proc/i2o/%s \n " , c - > name ) ;
pe = tmp ;
}
}
/**
* i2o_proc_fs_create - Create the i2o proc fs .
*
* Iterate over each I2O controller and create the entries for it .
*
* Returns 0 on success or negative error code on failure .
*/
static int __init i2o_proc_fs_create ( void )
{
struct i2o_controller * c ;
i2o_proc_dir_root = proc_mkdir ( " i2o " , NULL ) ;
if ( ! i2o_proc_dir_root )
return - 1 ;
i2o_proc_dir_root - > owner = THIS_MODULE ;
list_for_each_entry ( c , & i2o_controllers , list )
i2o_proc_iop_add ( i2o_proc_dir_root , c ) ;
return 0 ;
} ;
/**
* i2o_proc_fs_destroy - Cleanup the all i2o proc entries
*
* Iterate over each I2O controller and remove the entries for it .
*
* Returns 0 on success or negative error code on failure .
*/
static int __exit i2o_proc_fs_destroy ( void )
{
struct i2o_controller * c ;
list_for_each_entry ( c , & i2o_controllers , list )
i2o_proc_iop_remove ( i2o_proc_dir_root , c ) ;
remove_proc_entry ( " i2o " , NULL ) ;
return 0 ;
} ;
/**
* i2o_proc_init - Init function for procfs
*
* Registers Proc OSM and creates procfs entries .
*
* Returns 0 on success or negative error code on failure .
*/
static int __init i2o_proc_init ( void )
{
int rc ;
printk ( KERN_INFO OSM_DESCRIPTION " v " OSM_VERSION " \n " ) ;
rc = i2o_driver_register ( & i2o_proc_driver ) ;
if ( rc )
return rc ;
rc = i2o_proc_fs_create ( ) ;
if ( rc ) {
i2o_driver_unregister ( & i2o_proc_driver ) ;
return rc ;
}
return 0 ;
} ;
/**
* i2o_proc_exit - Exit function for procfs
*
* Unregisters Proc OSM and removes procfs entries .
*/
static void __exit i2o_proc_exit ( void )
{
i2o_driver_unregister ( & i2o_proc_driver ) ;
i2o_proc_fs_destroy ( ) ;
} ;
MODULE_AUTHOR ( " Deepak Saxena " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( OSM_DESCRIPTION ) ;
MODULE_VERSION ( OSM_VERSION ) ;
module_init ( i2o_proc_init ) ;
module_exit ( i2o_proc_exit ) ;