2005-04-16 15:20:36 -07:00
# ifndef _CSS_H
# define _CSS_H
2006-03-24 03:15:14 -08:00
# include <linux/mutex.h>
2005-04-16 15:20:36 -07:00
# include <linux/wait.h>
# include <linux/workqueue.h>
2007-04-27 16:01:28 +02:00
# include <linux/device.h>
2007-04-27 16:01:31 +02:00
# include <linux/types.h>
2005-04-16 15:20:36 -07:00
# include <asm/cio.h>
2007-04-27 16:01:31 +02:00
# include <asm/chpid.h>
2005-04-16 15:20:36 -07:00
2006-01-06 00:19:21 -08:00
# include "schid.h"
2005-04-16 15:20:36 -07:00
/*
* path grouping stuff
*/
# define SPID_FUNC_SINGLE_PATH 0x00
# define SPID_FUNC_MULTI_PATH 0x80
# define SPID_FUNC_ESTABLISH 0x00
# define SPID_FUNC_RESIGN 0x40
# define SPID_FUNC_DISBAND 0x20
# define SNID_STATE1_RESET 0
# define SNID_STATE1_UNGROUPED 2
# define SNID_STATE1_GROUPED 3
# define SNID_STATE2_NOT_RESVD 0
# define SNID_STATE2_RESVD_ELSE 2
# define SNID_STATE2_RESVD_SELF 3
# define SNID_STATE3_MULTI_PATH 1
# define SNID_STATE3_SINGLE_PATH 0
struct path_state {
__u8 state1 : 2 ; /* path state value 1 */
__u8 state2 : 2 ; /* path state value 2 */
__u8 state3 : 1 ; /* path state value 3 */
__u8 resvd : 3 ; /* reserved */
} __attribute__ ( ( packed ) ) ;
2006-01-06 00:19:23 -08:00
struct extended_cssid {
u8 version ;
u8 cssid ;
} __attribute__ ( ( packed ) ) ;
2005-04-16 15:20:36 -07:00
struct pgid {
union {
__u8 fc ; /* SPID function code */
struct path_state ps ; /* SNID path state */
2006-06-04 02:51:28 -07:00
} __attribute__ ( ( packed ) ) inf ;
2006-01-06 00:19:23 -08:00
union {
__u32 cpu_addr : 16 ; /* CPU address */
struct extended_cssid ext_cssid ;
2006-06-04 02:51:28 -07:00
} __attribute__ ( ( packed ) ) pgid_high ;
2005-04-16 15:20:36 -07:00
__u32 cpu_id : 24 ; /* CPU identification */
__u32 cpu_model : 16 ; /* CPU model */
__u32 tod_high ; /* high word TOD clock */
} __attribute__ ( ( packed ) ) ;
/*
* A css driver handles all subchannels of one type .
* Currently , we only care about I / O subchannels ( type 0 ) , these
* have a ccw_device connected to them .
*/
2006-01-11 10:56:22 +01:00
struct subchannel ;
2005-04-16 15:20:36 -07:00
struct css_driver {
2008-01-26 14:10:47 +01:00
struct module * owner ;
2005-04-16 15:20:36 -07:00
unsigned int subchannel_type ;
struct device_driver drv ;
2008-01-26 14:10:39 +01:00
void ( * irq ) ( struct subchannel * ) ;
int ( * notify ) ( struct subchannel * , int ) ;
void ( * verify ) ( struct subchannel * ) ;
void ( * termination ) ( struct subchannel * ) ;
2006-01-11 10:56:22 +01:00
int ( * probe ) ( struct subchannel * ) ;
int ( * remove ) ( struct subchannel * ) ;
void ( * shutdown ) ( struct subchannel * ) ;
2008-01-26 14:10:41 +01:00
const char * name ;
2005-04-16 15:20:36 -07:00
} ;
2008-01-26 14:10:38 +01:00
# define to_cssdriver(n) container_of(n, struct css_driver, drv)
2005-04-16 15:20:36 -07:00
/*
* all css_drivers have the css_bus_type
*/
extern struct bus_type css_bus_type ;
2008-01-26 14:10:41 +01:00
extern int css_driver_register ( struct css_driver * ) ;
extern void css_driver_unregister ( struct css_driver * ) ;
2006-07-12 16:39:50 +02:00
extern void css_sch_device_unregister ( struct subchannel * ) ;
2006-01-06 00:19:21 -08:00
extern struct subchannel * get_subchannel_by_schid ( struct subchannel_id ) ;
2005-04-16 15:20:36 -07:00
extern int css_init_done ;
2008-01-26 14:10:48 +01:00
int for_each_subchannel_staged ( int ( * fn_known ) ( struct subchannel * , void * ) ,
int ( * fn_unknown ) ( struct subchannel_id ,
void * ) , void * data ) ;
2006-01-06 00:19:22 -08:00
extern int for_each_subchannel ( int ( * fn ) ( struct subchannel_id , void * ) , void * ) ;
2007-04-27 16:01:34 +02:00
extern void css_process_crw ( int , int ) ;
2007-02-05 21:16:47 +01:00
extern void css_reiterate_subchannels ( void ) ;
2007-04-27 16:01:35 +02:00
void css_update_ssd_info ( struct subchannel * sch ) ;
2005-04-16 15:20:36 -07:00
2006-01-06 00:19:21 -08:00
# define __MAX_SUBCHANNEL 65535
2006-01-06 00:19:25 -08:00
# define __MAX_SSID 3
2006-01-06 00:19:23 -08:00
struct channel_subsystem {
u8 cssid ;
int valid ;
2006-01-14 13:21:03 -08:00
struct channel_path * chps [ __MAX_CHPID + 1 ] ;
2006-01-06 00:19:23 -08:00
struct device device ;
struct pgid global_pgid ;
2006-03-24 03:15:14 -08:00
struct mutex mutex ;
/* channel measurement related */
int cm_enabled ;
void * cub_addr1 ;
void * cub_addr2 ;
2006-12-08 15:54:28 +01:00
/* for orphaned ccw devices */
struct subchannel * pseudo_subchannel ;
2006-01-06 00:19:23 -08:00
} ;
# define to_css(dev) container_of(dev, struct channel_subsystem, device)
2005-04-16 15:20:36 -07:00
extern struct bus_type css_bus_type ;
2007-10-12 16:11:13 +02:00
extern struct channel_subsystem * channel_subsystems [ ] ;
2005-04-16 15:20:36 -07:00
/* Some helper functions for disconnected state. */
int device_is_disconnected ( struct subchannel * ) ;
void device_set_disconnected ( struct subchannel * ) ;
void device_trigger_reprobe ( struct subchannel * ) ;
/* Helper functions for vary on/off. */
int device_is_online ( struct subchannel * ) ;
2006-10-11 15:31:41 +02:00
void device_kill_io ( struct subchannel * ) ;
2006-12-04 15:41:04 +01:00
void device_set_intretry ( struct subchannel * sch ) ;
2006-12-04 15:41:01 +01:00
int device_trigger_verify ( struct subchannel * sch ) ;
2005-04-16 15:20:36 -07:00
/* Machine check helper function. */
void device_kill_pending_timer ( struct subchannel * ) ;
/* Helper functions to build lists for the slow path. */
2007-04-27 16:01:34 +02:00
void css_schedule_eval ( struct subchannel_id schid ) ;
void css_schedule_eval_all ( void ) ;
2005-04-16 15:20:36 -07:00
2006-12-08 15:54:28 +01:00
int sch_is_pseudo_sch ( struct subchannel * ) ;
2008-01-26 14:10:45 +01:00
struct schib ;
int css_sch_is_valid ( struct schib * ) ;
2006-12-08 15:54:28 +01:00
2005-04-16 15:20:36 -07:00
extern struct workqueue_struct * slow_path_wq ;
2006-12-08 15:54:21 +01:00
2006-12-08 15:55:57 +01:00
extern struct attribute_group * subch_attr_groups [ ] ;
2005-04-16 15:20:36 -07:00
# endif