ccwdev.h 7.7 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
2
 * Copyright IBM Corp. 2002, 2009
L
Linus Torvalds 已提交
3
 *
4
 * Author(s): Arnd Bergmann <arndb@de.ibm.com>
L
Linus Torvalds 已提交
5
 *
6
 * Interface for CCW device drivers
L
Linus Torvalds 已提交
7 8 9 10 11 12
 */
#ifndef _S390_CCWDEV_H_
#define _S390_CCWDEV_H_

#include <linux/device.h>
#include <linux/mod_devicetable.h>
13
#include <asm/fcx.h>
14
#include <asm/irq.h>
C
Cornelia Huck 已提交
15
#include <asm/schid.h>
L
Linus Torvalds 已提交
16 17 18 19

/* structs from asm/cio.h */
struct irb;
struct ccw1;
20
struct ccw_dev_id;
L
Linus Torvalds 已提交
21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67

/* simplified initializers for struct ccw_device:
 * CCW_DEVICE and CCW_DEVICE_DEVTYPE initialize one
 * entry in your MODULE_DEVICE_TABLE and set the match_flag correctly */
#define CCW_DEVICE(cu, cum) 						\
	.cu_type=(cu), .cu_model=(cum),					\
	.match_flags=(CCW_DEVICE_ID_MATCH_CU_TYPE			\
		   | (cum ? CCW_DEVICE_ID_MATCH_CU_MODEL : 0))

#define CCW_DEVICE_DEVTYPE(cu, cum, dev, devm)				\
	.cu_type=(cu), .cu_model=(cum), .dev_type=(dev), .dev_model=(devm),\
	.match_flags=CCW_DEVICE_ID_MATCH_CU_TYPE			\
		   | ((cum) ? CCW_DEVICE_ID_MATCH_CU_MODEL : 0) 	\
		   | CCW_DEVICE_ID_MATCH_DEVICE_TYPE			\
		   | ((devm) ? CCW_DEVICE_ID_MATCH_DEVICE_MODEL : 0)

/* scan through an array of device ids and return the first
 * entry that matches the device.
 *
 * the array must end with an entry containing zero match_flags
 */
static inline const struct ccw_device_id *
ccw_device_id_match(const struct ccw_device_id *array,
			const struct ccw_device_id *match)
{
	const struct ccw_device_id *id = array;

	for (id = array; id->match_flags; id++) {
		if ((id->match_flags & CCW_DEVICE_ID_MATCH_CU_TYPE)
		    && (id->cu_type != match->cu_type))
			continue;

		if ((id->match_flags & CCW_DEVICE_ID_MATCH_CU_MODEL)
		    && (id->cu_model != match->cu_model))
			continue;

		if ((id->match_flags & CCW_DEVICE_ID_MATCH_DEVICE_TYPE)
		    && (id->dev_type != match->dev_type))
			continue;

		if ((id->match_flags & CCW_DEVICE_ID_MATCH_DEVICE_MODEL)
		    && (id->dev_model != match->dev_model))
			continue;

		return id;
	}

H
Heiko Carstens 已提交
68
	return NULL;
L
Linus Torvalds 已提交
69 70
}

C
Cornelia Huck 已提交
71 72 73 74 75 76 77 78
/**
 * struct ccw_device - channel attached device
 * @ccwlock: pointer to device lock
 * @id: id of this device
 * @drv: ccw driver for this device
 * @dev: embedded device structure
 * @online: online status of device
 * @handler: interrupt handler
L
Linus Torvalds 已提交
79
 *
C
Cornelia Huck 已提交
80 81 82 83
 * @handler is a member of the device rather than the driver since a driver
 * can have different interrupt handlers for different ccw devices
 * (multi-subchannel drivers).
 */
L
Linus Torvalds 已提交
84 85
struct ccw_device {
	spinlock_t *ccwlock;
C
Cornelia Huck 已提交
86
/* private: */
L
Linus Torvalds 已提交
87
	struct ccw_device_private *private;	/* cio private information */
C
Cornelia Huck 已提交
88 89 90 91
/* public: */
	struct ccw_device_id id;
	struct ccw_driver *drv;
	struct device dev;
L
Linus Torvalds 已提交
92 93 94 95
	int online;
	void (*handler) (struct ccw_device *, unsigned long, struct irb *);
};

96 97 98 99 100 101 102 103 104 105
/*
 * Possible events used by the path_event notifier.
 */
#define PE_NONE				0x0
#define PE_PATH_GONE			0x1 /* A path is no longer available. */
#define PE_PATH_AVAILABLE		0x2 /* A path has become available and
					       was successfully verified. */
#define PE_PATHGROUP_ESTABLISHED	0x4 /* A pathgroup was reset and had
					       to be established again. */

106 107 108 109 110 111 112 113
/*
 * Possible CIO actions triggered by the unit check handler.
 */
enum uc_todo {
	UC_TODO_RETRY,
	UC_TODO_RETRY_ON_NEW_PATH,
	UC_TODO_STOP
};
L
Linus Torvalds 已提交
114

C
Cornelia Huck 已提交
115 116 117 118 119 120 121 122
/**
 * struct ccw driver - device driver for channel attached devices
 * @ids: ids supported by this driver
 * @probe: function called on probe
 * @remove: function called on remove
 * @set_online: called when setting device online
 * @set_offline: called when setting device offline
 * @notify: notify driver of device state changes
123
 * @path_event: notify driver of channel path events
124
 * @shutdown: called at device shutdown
125 126 127 128 129
 * @prepare: prepare for pm state transition
 * @complete: undo work done in @prepare
 * @freeze: callback for freezing during hibernation snapshotting
 * @thaw: undo work done in @freeze
 * @restore: callback for restoring after hibernation
130
 * @uc_handler: callback for unit check handler
C
Cornelia Huck 已提交
131
 * @driver: embedded device driver structure
132
 * @int_class: interruption class to use for accounting interrupts
C
Cornelia Huck 已提交
133
 */
L
Linus Torvalds 已提交
134
struct ccw_driver {
C
Cornelia Huck 已提交
135 136
	struct ccw_device_id *ids;
	int (*probe) (struct ccw_device *);
L
Linus Torvalds 已提交
137 138 139 140
	void (*remove) (struct ccw_device *);
	int (*set_online) (struct ccw_device *);
	int (*set_offline) (struct ccw_device *);
	int (*notify) (struct ccw_device *, int);
141
	void (*path_event) (struct ccw_device *, int *);
142
	void (*shutdown) (struct ccw_device *);
143 144 145 146 147
	int (*prepare) (struct ccw_device *);
	void (*complete) (struct ccw_device *);
	int (*freeze)(struct ccw_device *);
	int (*thaw) (struct ccw_device *);
	int (*restore)(struct ccw_device *);
148
	enum uc_todo (*uc_handler) (struct ccw_device *, struct irb *);
C
Cornelia Huck 已提交
149
	struct device_driver driver;
150
	enum interruption_class int_class;
L
Linus Torvalds 已提交
151 152 153 154 155 156 157 158 159 160 161 162 163
};

extern struct ccw_device *get_ccwdev_by_busid(struct ccw_driver *cdrv,
					      const char *bus_id);

/* devices drivers call these during module load and unload.
 * When a driver is registered, its probe method is called
 * when new devices for its type pop up */
extern int  ccw_driver_register   (struct ccw_driver *driver);
extern void ccw_driver_unregister (struct ccw_driver *driver);

struct ccw1;

164
extern int ccw_device_set_options_mask(struct ccw_device *, unsigned long);
L
Linus Torvalds 已提交
165
extern int ccw_device_set_options(struct ccw_device *, unsigned long);
166
extern void ccw_device_clear_options(struct ccw_device *, unsigned long);
167 168
int ccw_device_is_pathgroup(struct ccw_device *cdev);
int ccw_device_is_multipath(struct ccw_device *cdev);
L
Linus Torvalds 已提交
169 170 171 172 173 174 175 176 177

/* Allow for i/o completion notification after primary interrupt status. */
#define CCWDEV_EARLY_NOTIFICATION	0x0001
/* Report all interrupt conditions. */
#define CCWDEV_REPORT_ALL	 	0x0002
/* Try to perform path grouping. */
#define CCWDEV_DO_PATHGROUP             0x0004
/* Allow forced onlining of boxed devices. */
#define CCWDEV_ALLOW_FORCE              0x0008
178 179
/* Try to use multipath mode. */
#define CCWDEV_DO_MULTIPATH		0x0010
L
Linus Torvalds 已提交
180 181 182 183 184 185 186 187 188 189 190 191 192 193 194

extern int ccw_device_start(struct ccw_device *, struct ccw1 *,
			    unsigned long, __u8, unsigned long);
extern int ccw_device_start_timeout(struct ccw_device *, struct ccw1 *,
				    unsigned long, __u8, unsigned long, int);
extern int ccw_device_start_key(struct ccw_device *, struct ccw1 *,
				unsigned long, __u8, __u8, unsigned long);
extern int ccw_device_start_timeout_key(struct ccw_device *, struct ccw1 *,
					unsigned long, __u8, __u8,
					unsigned long, int);


extern int ccw_device_resume(struct ccw_device *);
extern int ccw_device_halt(struct ccw_device *, unsigned long);
extern int ccw_device_clear(struct ccw_device *, unsigned long);
195 196 197 198 199 200 201 202 203 204 205
int ccw_device_tm_start_key(struct ccw_device *cdev, struct tcw *tcw,
			    unsigned long intparm, u8 lpm, u8 key);
int ccw_device_tm_start_key(struct ccw_device *, struct tcw *,
			    unsigned long, u8, u8);
int ccw_device_tm_start_timeout_key(struct ccw_device *, struct tcw *,
			    unsigned long, u8, u8, int);
int ccw_device_tm_start(struct ccw_device *, struct tcw *,
			    unsigned long, u8);
int ccw_device_tm_start_timeout(struct ccw_device *, struct tcw *,
			    unsigned long, u8, int);
int ccw_device_tm_intrg(struct ccw_device *cdev);
L
Linus Torvalds 已提交
206

207 208
int ccw_device_get_mdc(struct ccw_device *cdev, u8 mask);

L
Linus Torvalds 已提交
209 210 211 212 213 214
extern int ccw_device_set_online(struct ccw_device *cdev);
extern int ccw_device_set_offline(struct ccw_device *cdev);


extern struct ciw *ccw_device_get_ciw(struct ccw_device *, __u32 cmd);
extern __u8 ccw_device_get_path_mask(struct ccw_device *);
215
extern void ccw_device_get_id(struct ccw_device *, struct ccw_dev_id *);
L
Linus Torvalds 已提交
216 217 218 219 220 221 222

#define get_ccwdev_lock(x) (x)->ccwlock

#define to_ccwdev(n) container_of(n, struct ccw_device, dev)
#define to_ccwdrv(n) container_of(n, struct ccw_driver, driver)

extern struct ccw_device *ccw_device_probe_console(void);
223
extern int ccw_device_force_console(void);
L
Linus Torvalds 已提交
224

M
Michael Ernst 已提交
225 226
int ccw_device_siosl(struct ccw_device *);

227 228
extern void ccw_device_get_schid(struct ccw_device *, struct subchannel_id *);

L
Linus Torvalds 已提交
229 230
extern void *ccw_device_get_chp_desc(struct ccw_device *, int);
#endif /* _S390_CCWDEV_H_ */