diff options
Diffstat (limited to 'include/scsi')
-rw-r--r-- | include/scsi/fc/fc_fcoe.h | 7 | ||||
-rw-r--r-- | include/scsi/fc_frame.h | 19 | ||||
-rw-r--r-- | include/scsi/libfc.h | 31 | ||||
-rw-r--r-- | include/scsi/libfcoe.h | 18 | ||||
-rw-r--r-- | include/scsi/libiscsi.h | 19 | ||||
-rw-r--r-- | include/scsi/osd_attributes.h | 327 | ||||
-rw-r--r-- | include/scsi/osd_initiator.h | 433 | ||||
-rw-r--r-- | include/scsi/osd_protocol.h | 579 | ||||
-rw-r--r-- | include/scsi/osd_sec.h | 45 | ||||
-rw-r--r-- | include/scsi/osd_sense.h | 260 | ||||
-rw-r--r-- | include/scsi/osd_types.h | 40 | ||||
-rw-r--r-- | include/scsi/scsi.h | 31 | ||||
-rw-r--r-- | include/scsi/scsi_cmnd.h | 15 | ||||
-rw-r--r-- | include/scsi/scsi_device.h | 10 | ||||
-rw-r--r-- | include/scsi/scsi_transport_iscsi.h | 4 |
15 files changed, 1747 insertions, 91 deletions
diff --git a/include/scsi/fc/fc_fcoe.h b/include/scsi/fc/fc_fcoe.h index f271d9cc0fc2..ccb3dbe90463 100644 --- a/include/scsi/fc/fc_fcoe.h +++ b/include/scsi/fc/fc_fcoe.h | |||
@@ -25,13 +25,6 @@ | |||
25 | */ | 25 | */ |
26 | 26 | ||
27 | /* | 27 | /* |
28 | * The FCoE ethertype eventually goes in net/if_ether.h. | ||
29 | */ | ||
30 | #ifndef ETH_P_FCOE | ||
31 | #define ETH_P_FCOE 0x8906 /* FCOE ether type */ | ||
32 | #endif | ||
33 | |||
34 | /* | ||
35 | * FC_FCOE_OUI hasn't been standardized yet. XXX TBD. | 28 | * FC_FCOE_OUI hasn't been standardized yet. XXX TBD. |
36 | */ | 29 | */ |
37 | #ifndef FC_FCOE_OUI | 30 | #ifndef FC_FCOE_OUI |
diff --git a/include/scsi/fc_frame.h b/include/scsi/fc_frame.h index 04d34a71355f..59511057cee0 100644 --- a/include/scsi/fc_frame.h +++ b/include/scsi/fc_frame.h | |||
@@ -54,8 +54,7 @@ | |||
54 | #define fr_eof(fp) (fr_cb(fp)->fr_eof) | 54 | #define fr_eof(fp) (fr_cb(fp)->fr_eof) |
55 | #define fr_flags(fp) (fr_cb(fp)->fr_flags) | 55 | #define fr_flags(fp) (fr_cb(fp)->fr_flags) |
56 | #define fr_max_payload(fp) (fr_cb(fp)->fr_max_payload) | 56 | #define fr_max_payload(fp) (fr_cb(fp)->fr_max_payload) |
57 | #define fr_cmd(fp) (fr_cb(fp)->fr_cmd) | 57 | #define fr_fsp(fp) (fr_cb(fp)->fr_fsp) |
58 | #define fr_dir(fp) (fr_cmd(fp)->sc_data_direction) | ||
59 | #define fr_crc(fp) (fr_cb(fp)->fr_crc) | 58 | #define fr_crc(fp) (fr_cb(fp)->fr_crc) |
60 | 59 | ||
61 | struct fc_frame { | 60 | struct fc_frame { |
@@ -66,7 +65,7 @@ struct fcoe_rcv_info { | |||
66 | struct packet_type *ptype; | 65 | struct packet_type *ptype; |
67 | struct fc_lport *fr_dev; /* transport layer private pointer */ | 66 | struct fc_lport *fr_dev; /* transport layer private pointer */ |
68 | struct fc_seq *fr_seq; /* for use with exchange manager */ | 67 | struct fc_seq *fr_seq; /* for use with exchange manager */ |
69 | struct scsi_cmnd *fr_cmd; /* for use of scsi command */ | 68 | struct fc_fcp_pkt *fr_fsp; /* for the corresponding fcp I/O */ |
70 | u32 fr_crc; | 69 | u32 fr_crc; |
71 | u16 fr_max_payload; /* max FC payload */ | 70 | u16 fr_max_payload; /* max FC payload */ |
72 | enum fc_sof fr_sof; /* start of frame delimiter */ | 71 | enum fc_sof fr_sof; /* start of frame delimiter */ |
@@ -218,20 +217,6 @@ static inline bool fc_frame_is_cmd(const struct fc_frame *fp) | |||
218 | return fc_frame_rctl(fp) == FC_RCTL_DD_UNSOL_CMD; | 217 | return fc_frame_rctl(fp) == FC_RCTL_DD_UNSOL_CMD; |
219 | } | 218 | } |
220 | 219 | ||
221 | static inline bool fc_frame_is_read(const struct fc_frame *fp) | ||
222 | { | ||
223 | if (fc_frame_is_cmd(fp) && fr_cmd(fp)) | ||
224 | return fr_dir(fp) == DMA_FROM_DEVICE; | ||
225 | return false; | ||
226 | } | ||
227 | |||
228 | static inline bool fc_frame_is_write(const struct fc_frame *fp) | ||
229 | { | ||
230 | if (fc_frame_is_cmd(fp) && fr_cmd(fp)) | ||
231 | return fr_dir(fp) == DMA_TO_DEVICE; | ||
232 | return false; | ||
233 | } | ||
234 | |||
235 | /* | 220 | /* |
236 | * Check for leaks. | 221 | * Check for leaks. |
237 | * Print the frame header of any currently allocated frame, assuming there | 222 | * Print the frame header of any currently allocated frame, assuming there |
diff --git a/include/scsi/libfc.h b/include/scsi/libfc.h index a2e126b86e3e..a70eafaad084 100644 --- a/include/scsi/libfc.h +++ b/include/scsi/libfc.h | |||
@@ -245,6 +245,7 @@ struct fc_fcp_pkt { | |||
245 | */ | 245 | */ |
246 | struct fcp_cmnd cdb_cmd; | 246 | struct fcp_cmnd cdb_cmd; |
247 | size_t xfer_len; | 247 | size_t xfer_len; |
248 | u16 xfer_ddp; /* this xfer is ddped */ | ||
248 | u32 xfer_contig_end; /* offset of end of contiguous xfer */ | 249 | u32 xfer_contig_end; /* offset of end of contiguous xfer */ |
249 | u16 max_payload; /* max payload size in bytes */ | 250 | u16 max_payload; /* max payload size in bytes */ |
250 | 251 | ||
@@ -267,6 +268,15 @@ struct fc_fcp_pkt { | |||
267 | u8 recov_retry; /* count of recovery retries */ | 268 | u8 recov_retry; /* count of recovery retries */ |
268 | struct fc_seq *recov_seq; /* sequence for REC or SRR */ | 269 | struct fc_seq *recov_seq; /* sequence for REC or SRR */ |
269 | }; | 270 | }; |
271 | /* | ||
272 | * FC_FCP HELPER FUNCTIONS | ||
273 | *****************************/ | ||
274 | static inline bool fc_fcp_is_read(const struct fc_fcp_pkt *fsp) | ||
275 | { | ||
276 | if (fsp && fsp->cmd) | ||
277 | return fsp->cmd->sc_data_direction == DMA_FROM_DEVICE; | ||
278 | return false; | ||
279 | } | ||
270 | 280 | ||
271 | /* | 281 | /* |
272 | * Structure and function definitions for managing Fibre Channel Exchanges | 282 | * Structure and function definitions for managing Fibre Channel Exchanges |
@@ -400,6 +410,21 @@ struct libfc_function_template { | |||
400 | void *arg, unsigned int timer_msec); | 410 | void *arg, unsigned int timer_msec); |
401 | 411 | ||
402 | /* | 412 | /* |
413 | * Sets up the DDP context for a given exchange id on the given | ||
414 | * scatterlist if LLD supports DDP for large receive. | ||
415 | * | ||
416 | * STATUS: OPTIONAL | ||
417 | */ | ||
418 | int (*ddp_setup)(struct fc_lport *lp, u16 xid, | ||
419 | struct scatterlist *sgl, unsigned int sgc); | ||
420 | /* | ||
421 | * Completes the DDP transfer and returns the length of data DDPed | ||
422 | * for the given exchange id. | ||
423 | * | ||
424 | * STATUS: OPTIONAL | ||
425 | */ | ||
426 | int (*ddp_done)(struct fc_lport *lp, u16 xid); | ||
427 | /* | ||
403 | * Send a frame using an existing sequence and exchange. | 428 | * Send a frame using an existing sequence and exchange. |
404 | * | 429 | * |
405 | * STATUS: OPTIONAL | 430 | * STATUS: OPTIONAL |
@@ -654,6 +679,7 @@ struct fc_lport { | |||
654 | u16 link_speed; | 679 | u16 link_speed; |
655 | u16 link_supported_speeds; | 680 | u16 link_supported_speeds; |
656 | u16 lro_xid; /* max xid for fcoe lro */ | 681 | u16 lro_xid; /* max xid for fcoe lro */ |
682 | unsigned int lso_max; /* max large send size */ | ||
657 | struct fc_ns_fts fcts; /* FC-4 type masks */ | 683 | struct fc_ns_fts fcts; /* FC-4 type masks */ |
658 | struct fc_els_rnid_gen rnid_gen; /* RNID information */ | 684 | struct fc_els_rnid_gen rnid_gen; /* RNID information */ |
659 | 685 | ||
@@ -821,6 +847,11 @@ int fc_change_queue_type(struct scsi_device *sdev, int tag_type); | |||
821 | void fc_fcp_destroy(struct fc_lport *); | 847 | void fc_fcp_destroy(struct fc_lport *); |
822 | 848 | ||
823 | /* | 849 | /* |
850 | * Set up direct-data placement for this I/O request | ||
851 | */ | ||
852 | void fc_fcp_ddp_setup(struct fc_fcp_pkt *fsp, u16 xid); | ||
853 | |||
854 | /* | ||
824 | * ELS/CT interface | 855 | * ELS/CT interface |
825 | *****************************/ | 856 | *****************************/ |
826 | /* | 857 | /* |
diff --git a/include/scsi/libfcoe.h b/include/scsi/libfcoe.h index 941818f29f59..c41f7d0c6efc 100644 --- a/include/scsi/libfcoe.h +++ b/include/scsi/libfcoe.h | |||
@@ -124,24 +124,6 @@ static inline u16 skb_fc_rxid(const struct sk_buff *skb) | |||
124 | return be16_to_cpu(skb_fc_header(skb)->fh_rx_id); | 124 | return be16_to_cpu(skb_fc_header(skb)->fh_rx_id); |
125 | } | 125 | } |
126 | 126 | ||
127 | /* FIXME - DMA_BIDIRECTIONAL ? */ | ||
128 | #define skb_cb(skb) ((struct fcoe_rcv_info *)&((skb)->cb[0])) | ||
129 | #define skb_cmd(skb) (skb_cb(skb)->fr_cmd) | ||
130 | #define skb_dir(skb) (skb_cmd(skb)->sc_data_direction) | ||
131 | static inline bool skb_fc_is_read(const struct sk_buff *skb) | ||
132 | { | ||
133 | if (skb_fc_is_cmd(skb) && skb_cmd(skb)) | ||
134 | return skb_dir(skb) == DMA_FROM_DEVICE; | ||
135 | return false; | ||
136 | } | ||
137 | |||
138 | static inline bool skb_fc_is_write(const struct sk_buff *skb) | ||
139 | { | ||
140 | if (skb_fc_is_cmd(skb) && skb_cmd(skb)) | ||
141 | return skb_dir(skb) == DMA_TO_DEVICE; | ||
142 | return false; | ||
143 | } | ||
144 | |||
145 | /* libfcoe funcs */ | 127 | /* libfcoe funcs */ |
146 | int fcoe_reset(struct Scsi_Host *shost); | 128 | int fcoe_reset(struct Scsi_Host *shost); |
147 | u64 fcoe_wwn_from_mac(unsigned char mac[MAX_ADDR_LEN], | 129 | u64 fcoe_wwn_from_mac(unsigned char mac[MAX_ADDR_LEN], |
diff --git a/include/scsi/libiscsi.h b/include/scsi/libiscsi.h index 7360e1916e75..7ffaed2f94dd 100644 --- a/include/scsi/libiscsi.h +++ b/include/scsi/libiscsi.h | |||
@@ -45,18 +45,10 @@ struct iscsi_session; | |||
45 | struct iscsi_nopin; | 45 | struct iscsi_nopin; |
46 | struct device; | 46 | struct device; |
47 | 47 | ||
48 | /* #define DEBUG_SCSI */ | ||
49 | #ifdef DEBUG_SCSI | ||
50 | #define debug_scsi(fmt...) printk(KERN_INFO "iscsi: " fmt) | ||
51 | #else | ||
52 | #define debug_scsi(fmt...) | ||
53 | #endif | ||
54 | |||
55 | #define ISCSI_DEF_XMIT_CMDS_MAX 128 /* must be power of 2 */ | 48 | #define ISCSI_DEF_XMIT_CMDS_MAX 128 /* must be power of 2 */ |
56 | #define ISCSI_MGMT_CMDS_MAX 15 | 49 | #define ISCSI_MGMT_CMDS_MAX 15 |
57 | 50 | ||
58 | #define ISCSI_DEF_CMD_PER_LUN 32 | 51 | #define ISCSI_DEF_CMD_PER_LUN 32 |
59 | #define ISCSI_MAX_CMD_PER_LUN 128 | ||
60 | 52 | ||
61 | /* Task Mgmt states */ | 53 | /* Task Mgmt states */ |
62 | enum { | 54 | enum { |
@@ -326,6 +318,9 @@ struct iscsi_host { | |||
326 | spinlock_t lock; | 318 | spinlock_t lock; |
327 | int num_sessions; | 319 | int num_sessions; |
328 | int state; | 320 | int state; |
321 | |||
322 | struct workqueue_struct *workq; | ||
323 | char workq_name[20]; | ||
329 | }; | 324 | }; |
330 | 325 | ||
331 | /* | 326 | /* |
@@ -351,7 +346,8 @@ extern int iscsi_host_get_param(struct Scsi_Host *shost, | |||
351 | enum iscsi_host_param param, char *buf); | 346 | enum iscsi_host_param param, char *buf); |
352 | extern int iscsi_host_add(struct Scsi_Host *shost, struct device *pdev); | 347 | extern int iscsi_host_add(struct Scsi_Host *shost, struct device *pdev); |
353 | extern struct Scsi_Host *iscsi_host_alloc(struct scsi_host_template *sht, | 348 | extern struct Scsi_Host *iscsi_host_alloc(struct scsi_host_template *sht, |
354 | int dd_data_size, uint16_t qdepth); | 349 | int dd_data_size, |
350 | bool xmit_can_sleep); | ||
355 | extern void iscsi_host_remove(struct Scsi_Host *shost); | 351 | extern void iscsi_host_remove(struct Scsi_Host *shost); |
356 | extern void iscsi_host_free(struct Scsi_Host *shost); | 352 | extern void iscsi_host_free(struct Scsi_Host *shost); |
357 | 353 | ||
@@ -382,11 +378,12 @@ extern void iscsi_conn_stop(struct iscsi_cls_conn *, int); | |||
382 | extern int iscsi_conn_bind(struct iscsi_cls_session *, struct iscsi_cls_conn *, | 378 | extern int iscsi_conn_bind(struct iscsi_cls_session *, struct iscsi_cls_conn *, |
383 | int); | 379 | int); |
384 | extern void iscsi_conn_failure(struct iscsi_conn *conn, enum iscsi_err err); | 380 | extern void iscsi_conn_failure(struct iscsi_conn *conn, enum iscsi_err err); |
385 | extern void iscsi_session_failure(struct iscsi_cls_session *cls_session, | 381 | extern void iscsi_session_failure(struct iscsi_session *session, |
386 | enum iscsi_err err); | 382 | enum iscsi_err err); |
387 | extern int iscsi_conn_get_param(struct iscsi_cls_conn *cls_conn, | 383 | extern int iscsi_conn_get_param(struct iscsi_cls_conn *cls_conn, |
388 | enum iscsi_param param, char *buf); | 384 | enum iscsi_param param, char *buf); |
389 | extern void iscsi_suspend_tx(struct iscsi_conn *conn); | 385 | extern void iscsi_suspend_tx(struct iscsi_conn *conn); |
386 | extern void iscsi_conn_queue_work(struct iscsi_conn *conn); | ||
390 | 387 | ||
391 | #define iscsi_conn_printk(prefix, _c, fmt, a...) \ | 388 | #define iscsi_conn_printk(prefix, _c, fmt, a...) \ |
392 | iscsi_cls_conn_printk(prefix, ((struct iscsi_conn *)_c)->cls_conn, \ | 389 | iscsi_cls_conn_printk(prefix, ((struct iscsi_conn *)_c)->cls_conn, \ |
diff --git a/include/scsi/osd_attributes.h b/include/scsi/osd_attributes.h new file mode 100644 index 000000000000..f888a6fda073 --- /dev/null +++ b/include/scsi/osd_attributes.h | |||
@@ -0,0 +1,327 @@ | |||
1 | #ifndef __OSD_ATTRIBUTES_H__ | ||
2 | #define __OSD_ATTRIBUTES_H__ | ||
3 | |||
4 | #include "osd_protocol.h" | ||
5 | |||
6 | /* | ||
7 | * Contains types and constants that define attribute pages and attribute | ||
8 | * numbers and their data types. | ||
9 | */ | ||
10 | |||
11 | #define ATTR_SET(pg, id, l, ptr) \ | ||
12 | { .attr_page = pg, .attr_id = id, .len = l, .val_ptr = ptr } | ||
13 | |||
14 | #define ATTR_DEF(pg, id, l) ATTR_SET(pg, id, l, NULL) | ||
15 | |||
16 | /* osd-r10 4.7.3 Attributes pages */ | ||
17 | enum { | ||
18 | OSD_APAGE_OBJECT_FIRST = 0x0, | ||
19 | OSD_APAGE_OBJECT_DIRECTORY = 0, | ||
20 | OSD_APAGE_OBJECT_INFORMATION = 1, | ||
21 | OSD_APAGE_OBJECT_QUOTAS = 2, | ||
22 | OSD_APAGE_OBJECT_TIMESTAMP = 3, | ||
23 | OSD_APAGE_OBJECT_COLLECTIONS = 4, | ||
24 | OSD_APAGE_OBJECT_SECURITY = 5, | ||
25 | OSD_APAGE_OBJECT_LAST = 0x2fffffff, | ||
26 | |||
27 | OSD_APAGE_PARTITION_FIRST = 0x30000000, | ||
28 | OSD_APAGE_PARTITION_DIRECTORY = OSD_APAGE_PARTITION_FIRST + 0, | ||
29 | OSD_APAGE_PARTITION_INFORMATION = OSD_APAGE_PARTITION_FIRST + 1, | ||
30 | OSD_APAGE_PARTITION_QUOTAS = OSD_APAGE_PARTITION_FIRST + 2, | ||
31 | OSD_APAGE_PARTITION_TIMESTAMP = OSD_APAGE_PARTITION_FIRST + 3, | ||
32 | OSD_APAGE_PARTITION_SECURITY = OSD_APAGE_PARTITION_FIRST + 5, | ||
33 | OSD_APAGE_PARTITION_LAST = 0x5FFFFFFF, | ||
34 | |||
35 | OSD_APAGE_COLLECTION_FIRST = 0x60000000, | ||
36 | OSD_APAGE_COLLECTION_DIRECTORY = OSD_APAGE_COLLECTION_FIRST + 0, | ||
37 | OSD_APAGE_COLLECTION_INFORMATION = OSD_APAGE_COLLECTION_FIRST + 1, | ||
38 | OSD_APAGE_COLLECTION_TIMESTAMP = OSD_APAGE_COLLECTION_FIRST + 3, | ||
39 | OSD_APAGE_COLLECTION_SECURITY = OSD_APAGE_COLLECTION_FIRST + 5, | ||
40 | OSD_APAGE_COLLECTION_LAST = 0x8FFFFFFF, | ||
41 | |||
42 | OSD_APAGE_ROOT_FIRST = 0x90000000, | ||
43 | OSD_APAGE_ROOT_DIRECTORY = OSD_APAGE_ROOT_FIRST + 0, | ||
44 | OSD_APAGE_ROOT_INFORMATION = OSD_APAGE_ROOT_FIRST + 1, | ||
45 | OSD_APAGE_ROOT_QUOTAS = OSD_APAGE_ROOT_FIRST + 2, | ||
46 | OSD_APAGE_ROOT_TIMESTAMP = OSD_APAGE_ROOT_FIRST + 3, | ||
47 | OSD_APAGE_ROOT_SECURITY = OSD_APAGE_ROOT_FIRST + 5, | ||
48 | OSD_APAGE_ROOT_LAST = 0xBFFFFFFF, | ||
49 | |||
50 | OSD_APAGE_RESERVED_TYPE_FIRST = 0xC0000000, | ||
51 | OSD_APAGE_RESERVED_TYPE_LAST = 0xEFFFFFFF, | ||
52 | |||
53 | OSD_APAGE_COMMON_FIRST = 0xF0000000, | ||
54 | OSD_APAGE_COMMON_LAST = 0xFFFFFFFE, | ||
55 | |||
56 | OSD_APAGE_REQUEST_ALL = 0xFFFFFFFF, | ||
57 | }; | ||
58 | |||
59 | /* subcategories of attr pages within each range above */ | ||
60 | enum { | ||
61 | OSD_APAGE_STD_FIRST = 0x0, | ||
62 | OSD_APAGE_STD_DIRECTORY = 0, | ||
63 | OSD_APAGE_STD_INFORMATION = 1, | ||
64 | OSD_APAGE_STD_QUOTAS = 2, | ||
65 | OSD_APAGE_STD_TIMESTAMP = 3, | ||
66 | OSD_APAGE_STD_COLLECTIONS = 4, | ||
67 | OSD_APAGE_STD_POLICY_SECURITY = 5, | ||
68 | OSD_APAGE_STD_LAST = 0x0000007F, | ||
69 | |||
70 | OSD_APAGE_RESERVED_FIRST = 0x00000080, | ||
71 | OSD_APAGE_RESERVED_LAST = 0x00007FFF, | ||
72 | |||
73 | OSD_APAGE_OTHER_STD_FIRST = 0x00008000, | ||
74 | OSD_APAGE_OTHER_STD_LAST = 0x0000EFFF, | ||
75 | |||
76 | OSD_APAGE_PUBLIC_FIRST = 0x0000F000, | ||
77 | OSD_APAGE_PUBLIC_LAST = 0x0000FFFF, | ||
78 | |||
79 | OSD_APAGE_APP_DEFINED_FIRST = 0x00010000, | ||
80 | OSD_APAGE_APP_DEFINED_LAST = 0x1FFFFFFF, | ||
81 | |||
82 | OSD_APAGE_VENDOR_SPECIFIC_FIRST = 0x20000000, | ||
83 | OSD_APAGE_VENDOR_SPECIFIC_LAST = 0x2FFFFFFF, | ||
84 | }; | ||
85 | |||
86 | enum { | ||
87 | OSD_ATTR_PAGE_IDENTIFICATION = 0, /* in all pages 40 bytes */ | ||
88 | }; | ||
89 | |||
90 | struct page_identification { | ||
91 | u8 vendor_identification[8]; | ||
92 | u8 page_identification[32]; | ||
93 | } __packed; | ||
94 | |||
95 | struct osd_attr_page_header { | ||
96 | __be32 page_number; | ||
97 | __be32 page_length; | ||
98 | } __packed; | ||
99 | |||
100 | /* 7.1.2.8 Root Information attributes page (OSD_APAGE_ROOT_INFORMATION) */ | ||
101 | enum { | ||
102 | OSD_ATTR_RI_OSD_SYSTEM_ID = 0x3, /* 20 */ | ||
103 | OSD_ATTR_RI_VENDOR_IDENTIFICATION = 0x4, /* 8 */ | ||
104 | OSD_ATTR_RI_PRODUCT_IDENTIFICATION = 0x5, /* 16 */ | ||
105 | OSD_ATTR_RI_PRODUCT_MODEL = 0x6, /* 32 */ | ||
106 | OSD_ATTR_RI_PRODUCT_REVISION_LEVEL = 0x7, /* 4 */ | ||
107 | OSD_ATTR_RI_PRODUCT_SERIAL_NUMBER = 0x8, /* variable */ | ||
108 | OSD_ATTR_RI_OSD_NAME = 0x9, /* variable */ | ||
109 | OSD_ATTR_RI_TOTAL_CAPACITY = 0x80, /* 8 */ | ||
110 | OSD_ATTR_RI_USED_CAPACITY = 0x81, /* 8 */ | ||
111 | OSD_ATTR_RI_NUMBER_OF_PARTITIONS = 0xC0, /* 8 */ | ||
112 | OSD_ATTR_RI_CLOCK = 0x100, /* 6 */ | ||
113 | }; | ||
114 | /* Root_Information_attributes_page does not have a get_page structure */ | ||
115 | |||
116 | /* 7.1.2.9 Partition Information attributes page | ||
117 | * (OSD_APAGE_PARTITION_INFORMATION) | ||
118 | */ | ||
119 | enum { | ||
120 | OSD_ATTR_PI_PARTITION_ID = 0x1, /* 8 */ | ||
121 | OSD_ATTR_PI_USERNAME = 0x9, /* variable */ | ||
122 | OSD_ATTR_PI_USED_CAPACITY = 0x81, /* 8 */ | ||
123 | OSD_ATTR_PI_NUMBER_OF_OBJECTS = 0xC1, /* 8 */ | ||
124 | }; | ||
125 | /* Partition Information attributes page does not have a get_page structure */ | ||
126 | |||
127 | /* 7.1.2.10 Collection Information attributes page | ||
128 | * (OSD_APAGE_COLLECTION_INFORMATION) | ||
129 | */ | ||
130 | enum { | ||
131 | OSD_ATTR_CI_PARTITION_ID = 0x1, /* 8 */ | ||
132 | OSD_ATTR_CI_COLLECTION_OBJECT_ID = 0x2, /* 8 */ | ||
133 | OSD_ATTR_CI_USERNAME = 0x9, /* variable */ | ||
134 | OSD_ATTR_CI_USED_CAPACITY = 0x81, /* 8 */ | ||
135 | }; | ||
136 | /* Collection Information attributes page does not have a get_page structure */ | ||
137 | |||
138 | /* 7.1.2.11 User Object Information attributes page | ||
139 | * (OSD_APAGE_OBJECT_INFORMATION) | ||
140 | */ | ||
141 | enum { | ||
142 | OSD_ATTR_OI_PARTITION_ID = 0x1, /* 8 */ | ||
143 | OSD_ATTR_OI_OBJECT_ID = 0x2, /* 8 */ | ||
144 | OSD_ATTR_OI_USERNAME = 0x9, /* variable */ | ||
145 | OSD_ATTR_OI_USED_CAPACITY = 0x81, /* 8 */ | ||
146 | OSD_ATTR_OI_LOGICAL_LENGTH = 0x82, /* 8 */ | ||
147 | }; | ||
148 | /* Object Information attributes page does not have a get_page structure */ | ||
149 | |||
150 | /* 7.1.2.12 Root Quotas attributes page (OSD_APAGE_ROOT_QUOTAS) */ | ||
151 | enum { | ||
152 | OSD_ATTR_RQ_DEFAULT_MAXIMUM_USER_OBJECT_LENGTH = 0x1, /* 8 */ | ||
153 | OSD_ATTR_RQ_PARTITION_CAPACITY_QUOTA = 0x10001, /* 8 */ | ||
154 | OSD_ATTR_RQ_PARTITION_OBJECT_COUNT = 0x10002, /* 8 */ | ||
155 | OSD_ATTR_RQ_PARTITION_COLLECTIONS_PER_USER_OBJECT = 0x10081, /* 4 */ | ||
156 | OSD_ATTR_RQ_PARTITION_COUNT = 0x20002, /* 8 */ | ||
157 | }; | ||
158 | |||
159 | struct Root_Quotas_attributes_page { | ||
160 | struct osd_attr_page_header hdr; /* id=R+2, size=0x24 */ | ||
161 | __be64 default_maximum_user_object_length; | ||
162 | __be64 partition_capacity_quota; | ||
163 | __be64 partition_object_count; | ||
164 | __be64 partition_collections_per_user_object; | ||
165 | __be64 partition_count; | ||
166 | } __packed; | ||
167 | |||
168 | /* 7.1.2.13 Partition Quotas attributes page (OSD_APAGE_PARTITION_QUOTAS)*/ | ||
169 | enum { | ||
170 | OSD_ATTR_PQ_DEFAULT_MAXIMUM_USER_OBJECT_LENGTH = 0x1, /* 8 */ | ||
171 | OSD_ATTR_PQ_CAPACITY_QUOTA = 0x10001, /* 8 */ | ||
172 | OSD_ATTR_PQ_OBJECT_COUNT = 0x10002, /* 8 */ | ||
173 | OSD_ATTR_PQ_COLLECTIONS_PER_USER_OBJECT = 0x10081, /* 4 */ | ||
174 | }; | ||
175 | |||
176 | struct Partition_Quotas_attributes_page { | ||
177 | struct osd_attr_page_header hdr; /* id=P+2, size=0x1C */ | ||
178 | __be64 default_maximum_user_object_length; | ||
179 | __be64 capacity_quota; | ||
180 | __be64 object_count; | ||
181 | __be64 collections_per_user_object; | ||
182 | } __packed; | ||
183 | |||
184 | /* 7.1.2.14 User Object Quotas attributes page (OSD_APAGE_OBJECT_QUOTAS) */ | ||
185 | enum { | ||
186 | OSD_ATTR_OQ_MAXIMUM_LENGTH = 0x1, /* 8 */ | ||
187 | }; | ||
188 | |||
189 | struct Object_Quotas_attributes_page { | ||
190 | struct osd_attr_page_header hdr; /* id=U+2, size=0x8 */ | ||
191 | __be64 maximum_length; | ||
192 | } __packed; | ||
193 | |||
194 | /* 7.1.2.15 Root Timestamps attributes page (OSD_APAGE_ROOT_TIMESTAMP) */ | ||
195 | enum { | ||
196 | OSD_ATTR_RT_ATTRIBUTES_ACCESSED_TIME = 0x2, /* 6 */ | ||
197 | OSD_ATTR_RT_ATTRIBUTES_MODIFIED_TIME = 0x3, /* 6 */ | ||
198 | OSD_ATTR_RT_TIMESTAMP_BYPASS = 0xFFFFFFFE, /* 1 */ | ||
199 | }; | ||
200 | |||
201 | struct root_timestamps_attributes_page { | ||
202 | struct osd_attr_page_header hdr; /* id=R+3, size=0xD */ | ||
203 | struct osd_timestamp attributes_accessed_time; | ||
204 | struct osd_timestamp attributes_modified_time; | ||
205 | u8 timestamp_bypass; | ||
206 | } __packed; | ||
207 | |||
208 | /* 7.1.2.16 Partition Timestamps attributes page | ||
209 | * (OSD_APAGE_PARTITION_TIMESTAMP) | ||
210 | */ | ||
211 | enum { | ||
212 | OSD_ATTR_PT_CREATED_TIME = 0x1, /* 6 */ | ||
213 | OSD_ATTR_PT_ATTRIBUTES_ACCESSED_TIME = 0x2, /* 6 */ | ||
214 | OSD_ATTR_PT_ATTRIBUTES_MODIFIED_TIME = 0x3, /* 6 */ | ||
215 | OSD_ATTR_PT_DATA_ACCESSED_TIME = 0x4, /* 6 */ | ||
216 | OSD_ATTR_PT_DATA_MODIFIED_TIME = 0x5, /* 6 */ | ||
217 | OSD_ATTR_PT_TIMESTAMP_BYPASS = 0xFFFFFFFE, /* 1 */ | ||
218 | }; | ||
219 | |||
220 | struct partition_timestamps_attributes_page { | ||
221 | struct osd_attr_page_header hdr; /* id=P+3, size=0x1F */ | ||
222 | struct osd_timestamp created_time; | ||
223 | struct osd_timestamp attributes_accessed_time; | ||
224 | struct osd_timestamp attributes_modified_time; | ||
225 | struct osd_timestamp data_accessed_time; | ||
226 | struct osd_timestamp data_modified_time; | ||
227 | u8 timestamp_bypass; | ||
228 | } __packed; | ||
229 | |||
230 | /* 7.1.2.17/18 Collection/Object Timestamps attributes page | ||
231 | * (OSD_APAGE_COLLECTION_TIMESTAMP/OSD_APAGE_OBJECT_TIMESTAMP) | ||
232 | */ | ||
233 | enum { | ||
234 | OSD_ATTR_OT_CREATED_TIME = 0x1, /* 6 */ | ||
235 | OSD_ATTR_OT_ATTRIBUTES_ACCESSED_TIME = 0x2, /* 6 */ | ||
236 | OSD_ATTR_OT_ATTRIBUTES_MODIFIED_TIME = 0x3, /* 6 */ | ||
237 | OSD_ATTR_OT_DATA_ACCESSED_TIME = 0x4, /* 6 */ | ||
238 | OSD_ATTR_OT_DATA_MODIFIED_TIME = 0x5, /* 6 */ | ||
239 | }; | ||
240 | |||
241 | /* same for collection */ | ||
242 | struct object_timestamps_attributes_page { | ||
243 | struct osd_attr_page_header hdr; /* id=C+3/3, size=0x1E */ | ||
244 | struct osd_timestamp created_time; | ||
245 | struct osd_timestamp attributes_accessed_time; | ||
246 | struct osd_timestamp attributes_modified_time; | ||
247 | struct osd_timestamp data_accessed_time; | ||
248 | struct osd_timestamp data_modified_time; | ||
249 | } __packed; | ||
250 | |||
251 | /* 7.1.2.19 Collections attributes page */ | ||
252 | /* TBD */ | ||
253 | |||
254 | /* 7.1.2.20 Root Policy/Security attributes page (OSD_APAGE_ROOT_SECURITY) */ | ||
255 | enum { | ||
256 | OSD_ATTR_RS_DEFAULT_SECURITY_METHOD = 0x1, /* 1 */ | ||
257 | OSD_ATTR_RS_OLDEST_VALID_NONCE_LIMIT = 0x2, /* 6 */ | ||
258 | OSD_ATTR_RS_NEWEST_VALID_NONCE_LIMIT = 0x3, /* 6 */ | ||
259 | OSD_ATTR_RS_PARTITION_DEFAULT_SECURITY_METHOD = 0x6, /* 1 */ | ||
260 | OSD_ATTR_RS_SUPPORTED_SECURITY_METHODS = 0x7, /* 2 */ | ||
261 | OSD_ATTR_RS_ADJUSTABLE_CLOCK = 0x9, /* 6 */ | ||
262 | OSD_ATTR_RS_MASTER_KEY_IDENTIFIER = 0x7FFD, /* 0 or 7 */ | ||
263 | OSD_ATTR_RS_ROOT_KEY_IDENTIFIER = 0x7FFE, /* 0 or 7 */ | ||
264 | OSD_ATTR_RS_SUPPORTED_INTEGRITY_ALGORITHM_0 = 0x80000000,/* 1,(x16)*/ | ||
265 | OSD_ATTR_RS_SUPPORTED_DH_GROUP_0 = 0x80000010,/* 1,(x16)*/ | ||
266 | }; | ||
267 | |||
268 | struct root_security_attributes_page { | ||
269 | struct osd_attr_page_header hdr; /* id=R+5, size=0x3F */ | ||
270 | u8 default_security_method; | ||
271 | u8 partition_default_security_method; | ||
272 | __be16 supported_security_methods; | ||
273 | u8 mki_valid_rki_valid; | ||
274 | struct osd_timestamp oldest_valid_nonce_limit; | ||
275 | struct osd_timestamp newest_valid_nonce_limit; | ||
276 | struct osd_timestamp adjustable_clock; | ||
277 | u8 master_key_identifier[32-25]; | ||
278 | u8 root_key_identifier[39-32]; | ||
279 | u8 supported_integrity_algorithm[16]; | ||
280 | u8 supported_dh_group[16]; | ||
281 | } __packed; | ||
282 | |||
283 | /* 7.1.2.21 Partition Policy/Security attributes page | ||
284 | * (OSD_APAGE_PARTITION_SECURITY) | ||
285 | */ | ||
286 | enum { | ||
287 | OSD_ATTR_PS_DEFAULT_SECURITY_METHOD = 0x1, /* 1 */ | ||
288 | OSD_ATTR_PS_OLDEST_VALID_NONCE = 0x2, /* 6 */ | ||
289 | OSD_ATTR_PS_NEWEST_VALID_NONCE = 0x3, /* 6 */ | ||
290 | OSD_ATTR_PS_REQUEST_NONCE_LIST_DEPTH = 0x4, /* 2 */ | ||
291 | OSD_ATTR_PS_FROZEN_WORKING_KEY_BIT_MASK = 0x5, /* 2 */ | ||
292 | OSD_ATTR_PS_PARTITION_KEY_IDENTIFIER = 0x7FFF, /* 0 or 7 */ | ||
293 | OSD_ATTR_PS_WORKING_KEY_IDENTIFIER_FIRST = 0x8000, /* 0 or 7 */ | ||
294 | OSD_ATTR_PS_WORKING_KEY_IDENTIFIER_LAST = 0x800F, /* 0 or 7 */ | ||
295 | OSD_ATTR_PS_POLICY_ACCESS_TAG = 0x40000001, /* 4 */ | ||
296 | OSD_ATTR_PS_USER_OBJECT_POLICY_ACCESS_TAG = 0x40000002, /* 4 */ | ||
297 | }; | ||
298 | |||
299 | struct partition_security_attributes_page { | ||
300 | struct osd_attr_page_header hdr; /* id=p+5, size=0x8f */ | ||
301 | u8 reserved[3]; | ||
302 | u8 default_security_method; | ||
303 | struct osd_timestamp oldest_valid_nonce; | ||
304 | struct osd_timestamp newest_valid_nonce; | ||
305 | __be16 request_nonce_list_depth; | ||
306 | __be16 frozen_working_key_bit_mask; | ||
307 | __be32 policy_access_tag; | ||
308 | __be32 user_object_policy_access_tag; | ||
309 | u8 pki_valid; | ||
310 | __be16 wki_00_0f_vld; | ||
311 | struct osd_key_identifier partition_key_identifier; | ||
312 | struct osd_key_identifier working_key_identifiers[16]; | ||
313 | } __packed; | ||
314 | |||
315 | /* 7.1.2.22/23 Collection/Object Policy-Security attributes page | ||
316 | * (OSD_APAGE_COLLECTION_SECURITY/OSD_APAGE_OBJECT_SECURITY) | ||
317 | */ | ||
318 | enum { | ||
319 | OSD_ATTR_OS_POLICY_ACCESS_TAG = 0x40000001, /* 4 */ | ||
320 | }; | ||
321 | |||
322 | struct object_security_attributes_page { | ||
323 | struct osd_attr_page_header hdr; /* id=C+5/5, size=4 */ | ||
324 | __be32 policy_access_tag; | ||
325 | } __packed; | ||
326 | |||
327 | #endif /*ndef __OSD_ATTRIBUTES_H__*/ | ||
diff --git a/include/scsi/osd_initiator.h b/include/scsi/osd_initiator.h new file mode 100644 index 000000000000..b24d9616eb46 --- /dev/null +++ b/include/scsi/osd_initiator.h | |||
@@ -0,0 +1,433 @@ | |||
1 | /* | ||
2 | * osd_initiator.h - OSD initiator API definition | ||
3 | * | ||
4 | * Copyright (C) 2008 Panasas Inc. All rights reserved. | ||
5 | * | ||
6 | * Authors: | ||
7 | * Boaz Harrosh <bharrosh@panasas.com> | ||
8 | * Benny Halevy <bhalevy@panasas.com> | ||
9 | * | ||
10 | * This program is free software; you can redistribute it and/or modify | ||
11 | * it under the terms of the GNU General Public License version 2 | ||
12 | * | ||
13 | */ | ||
14 | #ifndef __OSD_INITIATOR_H__ | ||
15 | #define __OSD_INITIATOR_H__ | ||
16 | |||
17 | #include "osd_protocol.h" | ||
18 | #include "osd_types.h" | ||
19 | |||
20 | #include <linux/blkdev.h> | ||
21 | |||
22 | /* Note: "NI" in comments below means "Not Implemented yet" */ | ||
23 | |||
24 | /* Configure of code: | ||
25 | * #undef if you *don't* want OSD v1 support in runtime. | ||
26 | * If #defined the initiator will dynamically configure to encode OSD v1 | ||
27 | * CDB's if the target is detected to be OSD v1 only. | ||
28 | * OSD v2 only commands, options, and attributes will be ignored if target | ||
29 | * is v1 only. | ||
30 | * If #defined will result in bigger/slower code (OK Slower maybe not) | ||
31 | * Q: Should this be CONFIG_SCSI_OSD_VER1_SUPPORT and set from Kconfig? | ||
32 | */ | ||
33 | #define OSD_VER1_SUPPORT y | ||
34 | |||
35 | enum osd_std_version { | ||
36 | OSD_VER_NONE = 0, | ||
37 | OSD_VER1 = 1, | ||
38 | OSD_VER2 = 2, | ||
39 | }; | ||
40 | |||
41 | /* | ||
42 | * Object-based Storage Device. | ||
43 | * This object represents an OSD device. | ||
44 | * It is not a full linux device in any way. It is only | ||
45 | * a place to hang resources associated with a Linux | ||
46 | * request Q and some default properties. | ||
47 | */ | ||
48 | struct osd_dev { | ||
49 | struct scsi_device *scsi_device; | ||
50 | unsigned def_timeout; | ||
51 | |||
52 | #ifdef OSD_VER1_SUPPORT | ||
53 | enum osd_std_version version; | ||
54 | #endif | ||
55 | }; | ||
56 | |||
57 | /* Retrieve/return osd_dev(s) for use by Kernel clients */ | ||
58 | struct osd_dev *osduld_path_lookup(const char *dev_name); /*Use IS_ERR/ERR_PTR*/ | ||
59 | void osduld_put_device(struct osd_dev *od); | ||
60 | |||
61 | /* Add/remove test ioctls from external modules */ | ||
62 | typedef int (do_test_fn)(struct osd_dev *od, unsigned cmd, unsigned long arg); | ||
63 | int osduld_register_test(unsigned ioctl, do_test_fn *do_test); | ||
64 | void osduld_unregister_test(unsigned ioctl); | ||
65 | |||
66 | /* These are called by uld at probe time */ | ||
67 | void osd_dev_init(struct osd_dev *od, struct scsi_device *scsi_device); | ||
68 | void osd_dev_fini(struct osd_dev *od); | ||
69 | |||
70 | /* some hi level device operations */ | ||
71 | int osd_auto_detect_ver(struct osd_dev *od, void *caps); /* GFP_KERNEL */ | ||
72 | |||
73 | /* we might want to use function vector in the future */ | ||
74 | static inline void osd_dev_set_ver(struct osd_dev *od, enum osd_std_version v) | ||
75 | { | ||
76 | #ifdef OSD_VER1_SUPPORT | ||
77 | od->version = v; | ||
78 | #endif | ||
79 | } | ||
80 | |||
81 | struct osd_request; | ||
82 | typedef void (osd_req_done_fn)(struct osd_request *or, void *private); | ||
83 | |||
84 | struct osd_request { | ||
85 | struct osd_cdb cdb; | ||
86 | struct osd_data_out_integrity_info out_data_integ; | ||
87 | struct osd_data_in_integrity_info in_data_integ; | ||
88 | |||
89 | struct osd_dev *osd_dev; | ||
90 | struct request *request; | ||
91 | |||
92 | struct _osd_req_data_segment { | ||
93 | void *buff; | ||
94 | unsigned alloc_size; /* 0 here means: don't call kfree */ | ||
95 | unsigned total_bytes; | ||
96 | } set_attr, enc_get_attr, get_attr; | ||
97 | |||
98 | struct _osd_io_info { | ||
99 | struct bio *bio; | ||
100 | u64 total_bytes; | ||
101 | struct request *req; | ||
102 | struct _osd_req_data_segment *last_seg; | ||
103 | u8 *pad_buff; | ||
104 | } out, in; | ||
105 | |||
106 | gfp_t alloc_flags; | ||
107 | unsigned timeout; | ||
108 | unsigned retries; | ||
109 | u8 sense[OSD_MAX_SENSE_LEN]; | ||
110 | enum osd_attributes_mode attributes_mode; | ||
111 | |||
112 | osd_req_done_fn *async_done; | ||
113 | void *async_private; | ||
114 | int async_error; | ||
115 | }; | ||
116 | |||
117 | /* OSD Version control */ | ||
118 | static inline bool osd_req_is_ver1(struct osd_request *or) | ||
119 | { | ||
120 | #ifdef OSD_VER1_SUPPORT | ||
121 | return or->osd_dev->version == OSD_VER1; | ||
122 | #else | ||
123 | return false; | ||
124 | #endif | ||
125 | } | ||
126 | |||
127 | /* | ||
128 | * How to use the osd library: | ||
129 | * | ||
130 | * osd_start_request | ||
131 | * Allocates a request. | ||
132 | * | ||
133 | * osd_req_* | ||
134 | * Call one of, to encode the desired operation. | ||
135 | * | ||
136 | * osd_add_{get,set}_attr | ||
137 | * Optionally add attributes to the CDB, list or page mode. | ||
138 | * | ||
139 | * osd_finalize_request | ||
140 | * Computes final data out/in offsets and signs the request, | ||
141 | * making it ready for execution. | ||
142 | * | ||
143 | * osd_execute_request | ||
144 | * May be called to execute it through the block layer. Other wise submit | ||
145 | * the associated block request in some other way. | ||
146 | * | ||
147 | * After execution: | ||
148 | * osd_req_decode_sense | ||
149 | * Decodes sense information to verify execution results. | ||
150 | * | ||
151 | * osd_req_decode_get_attr | ||
152 | * Retrieve osd_add_get_attr_list() values if used. | ||
153 | * | ||
154 | * osd_end_request | ||
155 | * Must be called to deallocate the request. | ||
156 | */ | ||
157 | |||
158 | /** | ||
159 | * osd_start_request - Allocate and initialize an osd_request | ||
160 | * | ||
161 | * @osd_dev: OSD device that holds the scsi-device and default values | ||
162 | * that the request is associated with. | ||
163 | * @gfp: The allocation flags to use for request allocation, and all | ||
164 | * subsequent allocations. This will be stored at | ||
165 | * osd_request->alloc_flags, can be changed by user later | ||
166 | * | ||
167 | * Allocate osd_request and initialize all members to the | ||
168 | * default/initial state. | ||
169 | */ | ||
170 | struct osd_request *osd_start_request(struct osd_dev *od, gfp_t gfp); | ||
171 | |||
172 | enum osd_req_options { | ||
173 | OSD_REQ_FUA = 0x08, /* Force Unit Access */ | ||
174 | OSD_REQ_DPO = 0x10, /* Disable Page Out */ | ||
175 | |||
176 | OSD_REQ_BYPASS_TIMESTAMPS = 0x80, | ||
177 | }; | ||
178 | |||
179 | /** | ||
180 | * osd_finalize_request - Sign request and prepare request for execution | ||
181 | * | ||
182 | * @or: osd_request to prepare | ||
183 | * @options: combination of osd_req_options bit flags or 0. | ||
184 | * @cap: A Pointer to an OSD_CAP_LEN bytes buffer that is received from | ||
185 | * The security manager as capabilities for this cdb. | ||
186 | * @cap_key: The cryptographic key used to sign the cdb/data. Can be null | ||
187 | * if NOSEC is used. | ||
188 | * | ||
189 | * The actual request and bios are only allocated here, so are the get_attr | ||
190 | * buffers that will receive the returned attributes. Copy's @cap to cdb. | ||
191 | * Sign the cdb/data with @cap_key. | ||
192 | */ | ||
193 | int osd_finalize_request(struct osd_request *or, | ||
194 | u8 options, const void *cap, const u8 *cap_key); | ||
195 | |||
196 | /** | ||
197 | * osd_execute_request - Execute the request synchronously through block-layer | ||
198 | * | ||
199 | * @or: osd_request to Executed | ||
200 | * | ||
201 | * Calls blk_execute_rq to q the command and waits for completion. | ||
202 | */ | ||
203 | int osd_execute_request(struct osd_request *or); | ||
204 | |||
205 | /** | ||
206 | * osd_execute_request_async - Execute the request without waitting. | ||
207 | * | ||
208 | * @or: - osd_request to Executed | ||
209 | * @done: (Optional) - Called at end of execution | ||
210 | * @private: - Will be passed to @done function | ||
211 | * | ||
212 | * Calls blk_execute_rq_nowait to queue the command. When execution is done | ||
213 | * optionally calls @done with @private as parameter. @or->async_error will | ||
214 | * have the return code | ||
215 | */ | ||
216 | int osd_execute_request_async(struct osd_request *or, | ||
217 | osd_req_done_fn *done, void *private); | ||
218 | |||
219 | /** | ||
220 | * osd_req_decode_sense_full - Decode sense information after execution. | ||
221 | * | ||
222 | * @or: - osd_request to examine | ||
223 | * @osi - Recievs a more detailed error report information (optional). | ||
224 | * @silent - Do not print to dmsg (Even if enabled) | ||
225 | * @bad_obj_list - Some commands act on multiple objects. Failed objects will | ||
226 | * be recieved here (optional) | ||
227 | * @max_obj - Size of @bad_obj_list. | ||
228 | * @bad_attr_list - List of failing attributes (optional) | ||
229 | * @max_attr - Size of @bad_attr_list. | ||
230 | * | ||
231 | * After execution, sense + return code can be analyzed using this function. The | ||
232 | * return code is the final disposition on the error. So it is possible that a | ||
233 | * CHECK_CONDITION was returned from target but this will return NO_ERROR, for | ||
234 | * example on recovered errors. All parameters are optional if caller does | ||
235 | * not need any returned information. | ||
236 | * Note: This function will also dump the error to dmsg according to settings | ||
237 | * of the SCSI_OSD_DPRINT_SENSE Kconfig value. Set @silent if you know the | ||
238 | * command would routinely fail, to not spam the dmsg file. | ||
239 | */ | ||
240 | struct osd_sense_info { | ||
241 | int key; /* one of enum scsi_sense_keys */ | ||
242 | int additional_code ; /* enum osd_additional_sense_codes */ | ||
243 | union { /* Sense specific information */ | ||
244 | u16 sense_info; | ||
245 | u16 cdb_field_offset; /* scsi_invalid_field_in_cdb */ | ||
246 | }; | ||
247 | union { /* Command specific information */ | ||
248 | u64 command_info; | ||
249 | }; | ||
250 | |||
251 | u32 not_initiated_command_functions; /* osd_command_functions_bits */ | ||
252 | u32 completed_command_functions; /* osd_command_functions_bits */ | ||
253 | struct osd_obj_id obj; | ||
254 | struct osd_attr attr; | ||
255 | }; | ||
256 | |||
257 | int osd_req_decode_sense_full(struct osd_request *or, | ||
258 | struct osd_sense_info *osi, bool silent, | ||
259 | struct osd_obj_id *bad_obj_list, int max_obj, | ||
260 | struct osd_attr *bad_attr_list, int max_attr); | ||
261 | |||
262 | static inline int osd_req_decode_sense(struct osd_request *or, | ||
263 | struct osd_sense_info *osi) | ||
264 | { | ||
265 | return osd_req_decode_sense_full(or, osi, false, NULL, 0, NULL, 0); | ||
266 | } | ||
267 | |||
268 | /** | ||
269 | * osd_end_request - return osd_request to free store | ||
270 | * | ||
271 | * @or: osd_request to free | ||
272 | * | ||
273 | * Deallocate all osd_request resources (struct req's, BIOs, buffers, etc.) | ||
274 | */ | ||
275 | void osd_end_request(struct osd_request *or); | ||
276 | |||
277 | /* | ||
278 | * CDB Encoding | ||
279 | * | ||
280 | * Note: call only one of the following methods. | ||
281 | */ | ||
282 | |||
283 | /* | ||
284 | * Device commands | ||
285 | */ | ||
286 | void osd_req_set_master_seed_xchg(struct osd_request *or, ...);/* NI */ | ||
287 | void osd_req_set_master_key(struct osd_request *or, ...);/* NI */ | ||
288 | |||
289 | void osd_req_format(struct osd_request *or, u64 tot_capacity); | ||
290 | |||
291 | /* list all partitions | ||
292 | * @list header must be initialized to zero on first run. | ||
293 | * | ||
294 | * Call osd_is_obj_list_done() to find if we got the complete list. | ||
295 | */ | ||
296 | int osd_req_list_dev_partitions(struct osd_request *or, | ||
297 | osd_id initial_id, struct osd_obj_id_list *list, unsigned nelem); | ||
298 | |||
299 | void osd_req_flush_obsd(struct osd_request *or, | ||
300 | enum osd_options_flush_scope_values); | ||
301 | |||
302 | void osd_req_perform_scsi_command(struct osd_request *or, | ||
303 | const u8 *cdb, ...);/* NI */ | ||
304 | void osd_req_task_management(struct osd_request *or, ...);/* NI */ | ||
305 | |||
306 | /* | ||
307 | * Partition commands | ||
308 | */ | ||
309 | void osd_req_create_partition(struct osd_request *or, osd_id partition); | ||
310 | void osd_req_remove_partition(struct osd_request *or, osd_id partition); | ||
311 | |||
312 | void osd_req_set_partition_key(struct osd_request *or, | ||
313 | osd_id partition, u8 new_key_id[OSD_CRYPTO_KEYID_SIZE], | ||
314 | u8 seed[OSD_CRYPTO_SEED_SIZE]);/* NI */ | ||
315 | |||
316 | /* list all collections in the partition | ||
317 | * @list header must be init to zero on first run. | ||
318 | * | ||
319 | * Call osd_is_obj_list_done() to find if we got the complete list. | ||
320 | */ | ||
321 | int osd_req_list_partition_collections(struct osd_request *or, | ||
322 | osd_id partition, osd_id initial_id, struct osd_obj_id_list *list, | ||
323 | unsigned nelem); | ||
324 | |||
325 | /* list all objects in the partition | ||
326 | * @list header must be init to zero on first run. | ||
327 | * | ||
328 | * Call osd_is_obj_list_done() to find if we got the complete list. | ||
329 | */ | ||
330 | int osd_req_list_partition_objects(struct osd_request *or, | ||
331 | osd_id partition, osd_id initial_id, struct osd_obj_id_list *list, | ||
332 | unsigned nelem); | ||
333 | |||
334 | void osd_req_flush_partition(struct osd_request *or, | ||
335 | osd_id partition, enum osd_options_flush_scope_values); | ||
336 | |||
337 | /* | ||
338 | * Collection commands | ||
339 | */ | ||
340 | void osd_req_create_collection(struct osd_request *or, | ||
341 | const struct osd_obj_id *);/* NI */ | ||
342 | void osd_req_remove_collection(struct osd_request *or, | ||
343 | const struct osd_obj_id *);/* NI */ | ||
344 | |||
345 | /* list all objects in the collection */ | ||
346 | int osd_req_list_collection_objects(struct osd_request *or, | ||
347 | const struct osd_obj_id *, osd_id initial_id, | ||
348 | struct osd_obj_id_list *list, unsigned nelem); | ||
349 | |||
350 | /* V2 only filtered list of objects in the collection */ | ||
351 | void osd_req_query(struct osd_request *or, ...);/* NI */ | ||
352 | |||
353 | void osd_req_flush_collection(struct osd_request *or, | ||
354 | const struct osd_obj_id *, enum osd_options_flush_scope_values); | ||
355 | |||
356 | void osd_req_get_member_attrs(struct osd_request *or, ...);/* V2-only NI */ | ||
357 | void osd_req_set_member_attrs(struct osd_request *or, ...);/* V2-only NI */ | ||
358 | |||
359 | /* | ||
360 | * Object commands | ||
361 | */ | ||
362 | void osd_req_create_object(struct osd_request *or, struct osd_obj_id *); | ||
363 | void osd_req_remove_object(struct osd_request *or, struct osd_obj_id *); | ||
364 | |||
365 | void osd_req_write(struct osd_request *or, | ||
366 | const struct osd_obj_id *, struct bio *data_out, u64 offset); | ||
367 | void osd_req_append(struct osd_request *or, | ||
368 | const struct osd_obj_id *, struct bio *data_out);/* NI */ | ||
369 | void osd_req_create_write(struct osd_request *or, | ||
370 | const struct osd_obj_id *, struct bio *data_out, u64 offset);/* NI */ | ||
371 | void osd_req_clear(struct osd_request *or, | ||
372 | const struct osd_obj_id *, u64 offset, u64 len);/* NI */ | ||
373 | void osd_req_punch(struct osd_request *or, | ||
374 | const struct osd_obj_id *, u64 offset, u64 len);/* V2-only NI */ | ||
375 | |||
376 | void osd_req_flush_object(struct osd_request *or, | ||
377 | const struct osd_obj_id *, enum osd_options_flush_scope_values, | ||
378 | /*V2*/ u64 offset, /*V2*/ u64 len); | ||
379 | |||
380 | void osd_req_read(struct osd_request *or, | ||
381 | const struct osd_obj_id *, struct bio *data_in, u64 offset); | ||
382 | |||
383 | /* | ||
384 | * Root/Partition/Collection/Object Attributes commands | ||
385 | */ | ||
386 | |||
387 | /* get before set */ | ||
388 | void osd_req_get_attributes(struct osd_request *or, const struct osd_obj_id *); | ||
389 | |||
390 | /* set before get */ | ||
391 | void osd_req_set_attributes(struct osd_request *or, const struct osd_obj_id *); | ||
392 | |||
393 | /* | ||
394 | * Attributes appended to most commands | ||
395 | */ | ||
396 | |||
397 | /* Attributes List mode (or V2 CDB) */ | ||
398 | /* | ||
399 | * TODO: In ver2 if at finalize time only one attr was set and no gets, | ||
400 | * then the Attributes CDB mode is used automatically to save IO. | ||
401 | */ | ||
402 | |||
403 | /* set a list of attributes. */ | ||
404 | int osd_req_add_set_attr_list(struct osd_request *or, | ||
405 | const struct osd_attr *, unsigned nelem); | ||
406 | |||
407 | /* get a list of attributes */ | ||
408 | int osd_req_add_get_attr_list(struct osd_request *or, | ||
409 | const struct osd_attr *, unsigned nelem); | ||
410 | |||
411 | /* | ||
412 | * Attributes list decoding | ||
413 | * Must be called after osd_request.request was executed | ||
414 | * It is called in a loop to decode the returned get_attr | ||
415 | * (see osd_add_get_attr) | ||
416 | */ | ||
417 | int osd_req_decode_get_attr_list(struct osd_request *or, | ||
418 | struct osd_attr *, int *nelem, void **iterator); | ||
419 | |||
420 | /* Attributes Page mode */ | ||
421 | |||
422 | /* | ||
423 | * Read an attribute page and optionally set one attribute | ||
424 | * | ||
425 | * Retrieves the attribute page directly to a user buffer. | ||
426 | * @attr_page_data shall stay valid until end of execution. | ||
427 | * See osd_attributes.h for common page structures | ||
428 | */ | ||
429 | int osd_req_add_get_attr_page(struct osd_request *or, | ||
430 | u32 page_id, void *attr_page_data, unsigned max_page_len, | ||
431 | const struct osd_attr *set_one); | ||
432 | |||
433 | #endif /* __OSD_LIB_H__ */ | ||
diff --git a/include/scsi/osd_protocol.h b/include/scsi/osd_protocol.h new file mode 100644 index 000000000000..cd3cbf764650 --- /dev/null +++ b/include/scsi/osd_protocol.h | |||
@@ -0,0 +1,579 @@ | |||
1 | /* | ||
2 | * osd_protocol.h - OSD T10 standard C definitions. | ||
3 | * | ||
4 | * Copyright (C) 2008 Panasas Inc. All rights reserved. | ||
5 | * | ||
6 | * Authors: | ||
7 | * Boaz Harrosh <bharrosh@panasas.com> | ||
8 | * Benny Halevy <bhalevy@panasas.com> | ||
9 | * | ||
10 | * This program is free software; you can redistribute it and/or modify | ||
11 | * it under the terms of the GNU General Public License version 2 | ||
12 | * | ||
13 | * This file contains types and constants that are defined by the protocol | ||
14 | * Note: All names and symbols are taken from the OSD standard's text. | ||
15 | */ | ||
16 | #ifndef __OSD_PROTOCOL_H__ | ||
17 | #define __OSD_PROTOCOL_H__ | ||
18 | |||
19 | #include <linux/types.h> | ||
20 | #include <asm/unaligned.h> | ||
21 | #include <scsi/scsi.h> | ||
22 | |||
23 | enum { | ||
24 | OSDv1_ADDITIONAL_CDB_LENGTH = 192, | ||
25 | OSDv1_TOTAL_CDB_LEN = OSDv1_ADDITIONAL_CDB_LENGTH + 8, | ||
26 | OSDv1_CAP_LEN = 80, | ||
27 | /* Latest supported version */ | ||
28 | /* OSD_ADDITIONAL_CDB_LENGTH = 216,*/ | ||
29 | OSD_ADDITIONAL_CDB_LENGTH = | ||
30 | OSDv1_ADDITIONAL_CDB_LENGTH, /* FIXME: Pete rev-001 sup */ | ||
31 | OSD_TOTAL_CDB_LEN = OSD_ADDITIONAL_CDB_LENGTH + 8, | ||
32 | /* OSD_CAP_LEN = 104,*/ | ||
33 | OSD_CAP_LEN = OSDv1_CAP_LEN,/* FIXME: Pete rev-001 sup */ | ||
34 | |||
35 | OSD_SYSTEMID_LEN = 20, | ||
36 | OSD_CRYPTO_KEYID_SIZE = 20, | ||
37 | /*FIXME: OSDv2_CRYPTO_KEYID_SIZE = 32,*/ | ||
38 | OSD_CRYPTO_SEED_SIZE = 4, | ||
39 | OSD_CRYPTO_NONCE_SIZE = 12, | ||
40 | OSD_MAX_SENSE_LEN = 252, /* from SPC-3 */ | ||
41 | |||
42 | OSD_PARTITION_FIRST_ID = 0x10000, | ||
43 | OSD_OBJECT_FIRST_ID = 0x10000, | ||
44 | }; | ||
45 | |||
46 | /* (osd-r10 5.2.4) | ||
47 | * osd2r03: 5.2.3 Caching control bits | ||
48 | */ | ||
49 | enum osd_options_byte { | ||
50 | OSD_CDB_FUA = 0x08, /* Force Unit Access */ | ||
51 | OSD_CDB_DPO = 0x10, /* Disable Page Out */ | ||
52 | }; | ||
53 | |||
54 | /* | ||
55 | * osd2r03: 5.2.5 Isolation. | ||
56 | * First 3 bits, V2-only. | ||
57 | * Also for attr 110h "default isolation method" at Root Information page | ||
58 | */ | ||
59 | enum osd_options_byte_isolation { | ||
60 | OSD_ISOLATION_DEFAULT = 0, | ||
61 | OSD_ISOLATION_NONE = 1, | ||
62 | OSD_ISOLATION_STRICT = 2, | ||
63 | OSD_ISOLATION_RANGE = 4, | ||
64 | OSD_ISOLATION_FUNCTIONAL = 5, | ||
65 | OSD_ISOLATION_VENDOR = 7, | ||
66 | }; | ||
67 | |||
68 | /* (osd-r10: 6.7) | ||
69 | * osd2r03: 6.8 FLUSH, FLUSH COLLECTION, FLUSH OSD, FLUSH PARTITION | ||
70 | */ | ||
71 | enum osd_options_flush_scope_values { | ||
72 | OSD_CDB_FLUSH_ALL = 0, | ||
73 | OSD_CDB_FLUSH_ATTR_ONLY = 1, | ||
74 | |||
75 | OSD_CDB_FLUSH_ALL_RECURSIVE = 2, | ||
76 | /* V2-only */ | ||
77 | OSD_CDB_FLUSH_ALL_RANGE = 2, | ||
78 | }; | ||
79 | |||
80 | /* osd2r03: 5.2.10 Timestamps control */ | ||
81 | enum { | ||
82 | OSD_CDB_NORMAL_TIMESTAMPS = 0, | ||
83 | OSD_CDB_BYPASS_TIMESTAMPS = 0x7f, | ||
84 | }; | ||
85 | |||
86 | /* (osd-r10: 5.2.2.1) | ||
87 | * osd2r03: 5.2.4.1 Get and set attributes CDB format selection | ||
88 | * 2 bits at second nibble of command_specific_options byte | ||
89 | */ | ||
90 | enum osd_attributes_mode { | ||
91 | /* V2-only */ | ||
92 | OSD_CDB_SET_ONE_ATTR = 0x10, | ||
93 | |||
94 | OSD_CDB_GET_ATTR_PAGE_SET_ONE = 0x20, | ||
95 | OSD_CDB_GET_SET_ATTR_LISTS = 0x30, | ||
96 | |||
97 | OSD_CDB_GET_SET_ATTR_MASK = 0x30, | ||
98 | }; | ||
99 | |||
100 | /* (osd-r10: 4.12.5) | ||
101 | * osd2r03: 4.14.5 Data-In and Data-Out buffer offsets | ||
102 | * byte offset = mantissa * (2^(exponent+8)) | ||
103 | * struct { | ||
104 | * unsigned mantissa: 28; | ||
105 | * int exponent: 04; | ||
106 | * } | ||
107 | */ | ||
108 | typedef __be32 __bitwise osd_cdb_offset; | ||
109 | |||
110 | enum { | ||
111 | OSD_OFFSET_UNUSED = 0xFFFFFFFF, | ||
112 | OSD_OFFSET_MAX_BITS = 28, | ||
113 | |||
114 | OSDv1_OFFSET_MIN_SHIFT = 8, | ||
115 | OSD_OFFSET_MIN_SHIFT = 3, | ||
116 | OSD_OFFSET_MAX_SHIFT = 16, | ||
117 | }; | ||
118 | |||
119 | /* Return the smallest allowed encoded offset that contains @offset. | ||
120 | * | ||
121 | * The actual encoded offset returned is @offset + *padding. | ||
122 | * (up to max_shift, non-inclusive) | ||
123 | */ | ||
124 | osd_cdb_offset __osd_encode_offset(u64 offset, unsigned *padding, | ||
125 | int min_shift, int max_shift); | ||
126 | |||
127 | /* Minimum alignment is 256 bytes | ||
128 | * Note: Seems from std v1 that exponent can be from 0+8 to 0xE+8 (inclusive) | ||
129 | * which is 8 to 23 but IBM code restricts it to 16, so be it. | ||
130 | */ | ||
131 | static inline osd_cdb_offset osd_encode_offset_v1(u64 offset, unsigned *padding) | ||
132 | { | ||
133 | return __osd_encode_offset(offset, padding, | ||
134 | OSDv1_OFFSET_MIN_SHIFT, OSD_OFFSET_MAX_SHIFT); | ||
135 | } | ||
136 | |||
137 | /* Minimum 8 bytes alignment | ||
138 | * Same as v1 but since exponent can be signed than a less than | ||
139 | * 256 alignment can be reached with small offsets (<2GB) | ||
140 | */ | ||
141 | static inline osd_cdb_offset osd_encode_offset_v2(u64 offset, unsigned *padding) | ||
142 | { | ||
143 | return __osd_encode_offset(offset, padding, | ||
144 | OSD_OFFSET_MIN_SHIFT, OSD_OFFSET_MAX_SHIFT); | ||
145 | } | ||
146 | |||
147 | /* osd2r03: 5.2.1 Overview */ | ||
148 | struct osd_cdb_head { | ||
149 | struct scsi_varlen_cdb_hdr varlen_cdb; | ||
150 | /*10*/ u8 options; | ||
151 | u8 command_specific_options; | ||
152 | u8 timestamp_control; | ||
153 | /*13*/ u8 reserved1[3]; | ||
154 | /*16*/ __be64 partition; | ||
155 | /*24*/ __be64 object; | ||
156 | /*32*/ union { /* V1 vs V2 alignment differences */ | ||
157 | struct __osdv1_cdb_addr_len { | ||
158 | /*32*/ __be32 list_identifier;/* Rarely used */ | ||
159 | /*36*/ __be64 length; | ||
160 | /*44*/ __be64 start_address; | ||
161 | } __packed v1; | ||
162 | |||
163 | struct __osdv2_cdb_addr_len { | ||
164 | /* called allocation_length in some commands */ | ||
165 | /*32*/ __be64 length; | ||
166 | /*40*/ __be64 start_address; | ||
167 | /*48*/ __be32 list_identifier;/* Rarely used */ | ||
168 | } __packed v2; | ||
169 | }; | ||
170 | /*52*/ union { /* selected attributes mode Page/List/Single */ | ||
171 | struct osd_attributes_page_mode { | ||
172 | /*52*/ __be32 get_attr_page; | ||
173 | /*56*/ __be32 get_attr_alloc_length; | ||
174 | /*60*/ osd_cdb_offset get_attr_offset; | ||
175 | |||
176 | /*64*/ __be32 set_attr_page; | ||
177 | /*68*/ __be32 set_attr_id; | ||
178 | /*72*/ __be32 set_attr_length; | ||
179 | /*76*/ osd_cdb_offset set_attr_offset; | ||
180 | /*80*/ } __packed attrs_page; | ||
181 | |||
182 | struct osd_attributes_list_mode { | ||
183 | /*52*/ __be32 get_attr_desc_bytes; | ||
184 | /*56*/ osd_cdb_offset get_attr_desc_offset; | ||
185 | |||
186 | /*60*/ __be32 get_attr_alloc_length; | ||
187 | /*64*/ osd_cdb_offset get_attr_offset; | ||
188 | |||
189 | /*68*/ __be32 set_attr_bytes; | ||
190 | /*72*/ osd_cdb_offset set_attr_offset; | ||
191 | __be32 not_used; | ||
192 | /*80*/ } __packed attrs_list; | ||
193 | |||
194 | /* osd2r03:5.2.4.2 Set one attribute value using CDB fields */ | ||
195 | struct osd_attributes_cdb_mode { | ||
196 | /*52*/ __be32 set_attr_page; | ||
197 | /*56*/ __be32 set_attr_id; | ||
198 | /*60*/ __be16 set_attr_len; | ||
199 | /*62*/ u8 set_attr_val[18]; | ||
200 | /*80*/ } __packed attrs_cdb; | ||
201 | /*52*/ u8 get_set_attributes_parameters[28]; | ||
202 | }; | ||
203 | } __packed; | ||
204 | /*80*/ | ||
205 | |||
206 | /*160 v1*/ | ||
207 | /*184 v2*/ | ||
208 | struct osd_security_parameters { | ||
209 | /*160*/u8 integrity_check_value[OSD_CRYPTO_KEYID_SIZE]; | ||
210 | /*180*/u8 request_nonce[OSD_CRYPTO_NONCE_SIZE]; | ||
211 | /*192*/osd_cdb_offset data_in_integrity_check_offset; | ||
212 | /*196*/osd_cdb_offset data_out_integrity_check_offset; | ||
213 | } __packed; | ||
214 | /*200 v1*/ | ||
215 | /*224 v2*/ | ||
216 | |||
217 | /* FIXME: osdv2_security_parameters */ | ||
218 | |||
219 | struct osdv1_cdb { | ||
220 | struct osd_cdb_head h; | ||
221 | u8 caps[OSDv1_CAP_LEN]; | ||
222 | struct osd_security_parameters sec_params; | ||
223 | } __packed; | ||
224 | |||
225 | struct osdv2_cdb { | ||
226 | struct osd_cdb_head h; | ||
227 | u8 caps[OSD_CAP_LEN]; | ||
228 | struct osd_security_parameters sec_params; | ||
229 | /* FIXME: osdv2_security_parameters */ | ||
230 | } __packed; | ||
231 | |||
232 | struct osd_cdb { | ||
233 | union { | ||
234 | struct osdv1_cdb v1; | ||
235 | struct osdv2_cdb v2; | ||
236 | u8 buff[OSD_TOTAL_CDB_LEN]; | ||
237 | }; | ||
238 | } __packed; | ||
239 | |||
240 | static inline struct osd_cdb_head *osd_cdb_head(struct osd_cdb *ocdb) | ||
241 | { | ||
242 | return (struct osd_cdb_head *)ocdb->buff; | ||
243 | } | ||
244 | |||
245 | /* define both version actions | ||
246 | * Ex name = FORMAT_OSD we have OSD_ACT_FORMAT_OSD && OSDv1_ACT_FORMAT_OSD | ||
247 | */ | ||
248 | #define OSD_ACT___(Name, Num) \ | ||
249 | OSD_ACT_##Name = __constant_cpu_to_be16(0x8880 + Num), \ | ||
250 | OSDv1_ACT_##Name = __constant_cpu_to_be16(0x8800 + Num), | ||
251 | |||
252 | /* V2 only actions */ | ||
253 | #define OSD_ACT_V2(Name, Num) \ | ||
254 | OSD_ACT_##Name = __constant_cpu_to_be16(0x8880 + Num), | ||
255 | |||
256 | #define OSD_ACT_V1_V2(Name, Num1, Num2) \ | ||
257 | OSD_ACT_##Name = __constant_cpu_to_be16(Num2), \ | ||
258 | OSDv1_ACT_##Name = __constant_cpu_to_be16(Num1), | ||
259 | |||
260 | enum osd_service_actions { | ||
261 | OSD_ACT_V2(OBJECT_STRUCTURE_CHECK, 0x00) | ||
262 | OSD_ACT___(FORMAT_OSD, 0x01) | ||
263 | OSD_ACT___(CREATE, 0x02) | ||
264 | OSD_ACT___(LIST, 0x03) | ||
265 | OSD_ACT_V2(PUNCH, 0x04) | ||
266 | OSD_ACT___(READ, 0x05) | ||
267 | OSD_ACT___(WRITE, 0x06) | ||
268 | OSD_ACT___(APPEND, 0x07) | ||
269 | OSD_ACT___(FLUSH, 0x08) | ||
270 | OSD_ACT_V2(CLEAR, 0x09) | ||
271 | OSD_ACT___(REMOVE, 0x0A) | ||
272 | OSD_ACT___(CREATE_PARTITION, 0x0B) | ||
273 | OSD_ACT___(REMOVE_PARTITION, 0x0C) | ||
274 | OSD_ACT___(GET_ATTRIBUTES, 0x0E) | ||
275 | OSD_ACT___(SET_ATTRIBUTES, 0x0F) | ||
276 | OSD_ACT___(CREATE_AND_WRITE, 0x12) | ||
277 | OSD_ACT___(CREATE_COLLECTION, 0x15) | ||
278 | OSD_ACT___(REMOVE_COLLECTION, 0x16) | ||
279 | OSD_ACT___(LIST_COLLECTION, 0x17) | ||
280 | OSD_ACT___(SET_KEY, 0x18) | ||
281 | OSD_ACT___(SET_MASTER_KEY, 0x19) | ||
282 | OSD_ACT___(FLUSH_COLLECTION, 0x1A) | ||
283 | OSD_ACT___(FLUSH_PARTITION, 0x1B) | ||
284 | OSD_ACT___(FLUSH_OSD, 0x1C) | ||
285 | |||
286 | OSD_ACT_V2(QUERY, 0x20) | ||
287 | OSD_ACT_V2(REMOVE_MEMBER_OBJECTS, 0x21) | ||
288 | OSD_ACT_V2(GET_MEMBER_ATTRIBUTES, 0x22) | ||
289 | OSD_ACT_V2(SET_MEMBER_ATTRIBUTES, 0x23) | ||
290 | OSD_ACT_V2(READ_MAP, 0x31) | ||
291 | |||
292 | OSD_ACT_V1_V2(PERFORM_SCSI_COMMAND, 0x8F7E, 0x8F7C) | ||
293 | OSD_ACT_V1_V2(SCSI_TASK_MANAGEMENT, 0x8F7F, 0x8F7D) | ||
294 | /* 0x8F80 to 0x8FFF are Vendor specific */ | ||
295 | }; | ||
296 | |||
297 | /* osd2r03: 7.1.3.2 List entry format for retrieving attributes */ | ||
298 | struct osd_attributes_list_attrid { | ||
299 | __be32 attr_page; | ||
300 | __be32 attr_id; | ||
301 | } __packed; | ||
302 | |||
303 | /* | ||
304 | * osd2r03: 7.1.3.3 List entry format for retrieved attributes and | ||
305 | * for setting attributes | ||
306 | * NOTE: v2 is 8-bytes aligned, v1 is not aligned. | ||
307 | */ | ||
308 | struct osd_attributes_list_element { | ||
309 | __be32 attr_page; | ||
310 | __be32 attr_id; | ||
311 | __be16 attr_bytes; | ||
312 | u8 attr_val[0]; | ||
313 | } __packed; | ||
314 | |||
315 | enum { | ||
316 | OSDv1_ATTRIBUTES_ELEM_ALIGN = 1, | ||
317 | OSD_ATTRIBUTES_ELEM_ALIGN = 8, | ||
318 | }; | ||
319 | |||
320 | enum { | ||
321 | OSD_ATTR_LIST_ALL_PAGES = 0xFFFFFFFF, | ||
322 | OSD_ATTR_LIST_ALL_IN_PAGE = 0xFFFFFFFF, | ||
323 | }; | ||
324 | |||
325 | static inline unsigned osdv1_attr_list_elem_size(unsigned len) | ||
326 | { | ||
327 | return ALIGN(len + sizeof(struct osd_attributes_list_element), | ||
328 | OSDv1_ATTRIBUTES_ELEM_ALIGN); | ||
329 | } | ||
330 | |||
331 | static inline unsigned osdv2_attr_list_elem_size(unsigned len) | ||
332 | { | ||
333 | return ALIGN(len + sizeof(struct osd_attributes_list_element), | ||
334 | OSD_ATTRIBUTES_ELEM_ALIGN); | ||
335 | } | ||
336 | |||
337 | /* | ||
338 | * osd2r03: 7.1.3 OSD attributes lists (Table 184) — List type values | ||
339 | */ | ||
340 | enum osd_attr_list_types { | ||
341 | OSD_ATTR_LIST_GET = 0x1, /* descriptors only */ | ||
342 | OSD_ATTR_LIST_SET_RETRIEVE = 0x9, /*descriptors/values variable-length*/ | ||
343 | OSD_V2_ATTR_LIST_MULTIPLE = 0xE, /* ver2, Multiple Objects lists*/ | ||
344 | OSD_V1_ATTR_LIST_CREATE_MULTIPLE = 0xF,/*ver1, used by create_multple*/ | ||
345 | }; | ||
346 | |||
347 | /* osd2r03: 7.1.3.4 Multi-object retrieved attributes format */ | ||
348 | struct osd_attributes_list_multi_header { | ||
349 | __be64 object_id; | ||
350 | u8 object_type; /* object_type enum below */ | ||
351 | u8 reserved[5]; | ||
352 | __be16 list_bytes; | ||
353 | /* followed by struct osd_attributes_list_element's */ | ||
354 | }; | ||
355 | |||
356 | struct osdv1_attributes_list_header { | ||
357 | u8 type; /* low 4-bit only */ | ||
358 | u8 pad; | ||
359 | __be16 list_bytes; /* Initiator shall set to Zero. Only set by target */ | ||
360 | /* | ||
361 | * type=9 followed by struct osd_attributes_list_element's | ||
362 | * type=E followed by struct osd_attributes_list_multi_header's | ||
363 | */ | ||
364 | } __packed; | ||
365 | |||
366 | static inline unsigned osdv1_list_size(struct osdv1_attributes_list_header *h) | ||
367 | { | ||
368 | return be16_to_cpu(h->list_bytes); | ||
369 | } | ||
370 | |||
371 | struct osdv2_attributes_list_header { | ||
372 | u8 type; /* lower 4-bits only */ | ||
373 | u8 pad[3]; | ||
374 | /*4*/ __be32 list_bytes; /* Initiator shall set to zero. Only set by target */ | ||
375 | /* | ||
376 | * type=9 followed by struct osd_attributes_list_element's | ||
377 | * type=E followed by struct osd_attributes_list_multi_header's | ||
378 | */ | ||
379 | } __packed; | ||
380 | |||
381 | static inline unsigned osdv2_list_size(struct osdv2_attributes_list_header *h) | ||
382 | { | ||
383 | return be32_to_cpu(h->list_bytes); | ||
384 | } | ||
385 | |||
386 | /* (osd-r10 6.13) | ||
387 | * osd2r03: 6.15 LIST (Table 79) LIST command parameter data. | ||
388 | * for root_lstchg below | ||
389 | */ | ||
390 | enum { | ||
391 | OSD_OBJ_ID_LIST_PAR = 0x1, /* V1-only. Not used in V2 */ | ||
392 | OSD_OBJ_ID_LIST_LSTCHG = 0x2, | ||
393 | }; | ||
394 | |||
395 | /* | ||
396 | * osd2r03: 6.15.2 LIST command parameter data | ||
397 | * (Also for LIST COLLECTION) | ||
398 | */ | ||
399 | struct osd_obj_id_list { | ||
400 | __be64 list_bytes; /* bytes in list excluding list_bytes (-8) */ | ||
401 | __be64 continuation_id; | ||
402 | __be32 list_identifier; | ||
403 | u8 pad[3]; | ||
404 | u8 root_lstchg; | ||
405 | __be64 object_ids[0]; | ||
406 | } __packed; | ||
407 | |||
408 | static inline bool osd_is_obj_list_done(struct osd_obj_id_list *list, | ||
409 | bool *is_changed) | ||
410 | { | ||
411 | *is_changed = (0 != (list->root_lstchg & OSD_OBJ_ID_LIST_LSTCHG)); | ||
412 | return 0 != list->continuation_id; | ||
413 | } | ||
414 | |||
415 | /* | ||
416 | * osd2r03: 4.12.4.5 The ALLDATA security method | ||
417 | */ | ||
418 | struct osd_data_out_integrity_info { | ||
419 | __be64 data_bytes; | ||
420 | __be64 set_attributes_bytes; | ||
421 | __be64 get_attributes_bytes; | ||
422 | __be64 integrity_check_value; | ||
423 | } __packed; | ||
424 | |||
425 | struct osd_data_in_integrity_info { | ||
426 | __be64 data_bytes; | ||
427 | __be64 retrieved_attributes_bytes; | ||
428 | __be64 integrity_check_value; | ||
429 | } __packed; | ||
430 | |||
431 | struct osd_timestamp { | ||
432 | u8 time[6]; /* number of milliseconds since 1/1/1970 UT (big endian) */ | ||
433 | } __packed; | ||
434 | /* FIXME: define helper functions to convert to/from osd time format */ | ||
435 | |||
436 | /* | ||
437 | * Capability & Security definitions | ||
438 | * osd2r03: 4.11.2.2 Capability format | ||
439 | * osd2r03: 5.2.8 Security parameters | ||
440 | */ | ||
441 | |||
442 | struct osd_key_identifier { | ||
443 | u8 id[7]; /* if you know why 7 please email bharrosh@panasas.com */ | ||
444 | } __packed; | ||
445 | |||
446 | /* for osd_capability.format */ | ||
447 | enum { | ||
448 | OSD_SEC_CAP_FORMAT_NO_CAPS = 0, | ||
449 | OSD_SEC_CAP_FORMAT_VER1 = 1, | ||
450 | OSD_SEC_CAP_FORMAT_VER2 = 2, | ||
451 | }; | ||
452 | |||
453 | /* security_method */ | ||
454 | enum { | ||
455 | OSD_SEC_NOSEC = 0, | ||
456 | OSD_SEC_CAPKEY = 1, | ||
457 | OSD_SEC_CMDRSP = 2, | ||
458 | OSD_SEC_ALLDATA = 3, | ||
459 | }; | ||
460 | |||
461 | enum object_type { | ||
462 | OSD_SEC_OBJ_ROOT = 0x1, | ||
463 | OSD_SEC_OBJ_PARTITION = 0x2, | ||
464 | OSD_SEC_OBJ_COLLECTION = 0x40, | ||
465 | OSD_SEC_OBJ_USER = 0x80, | ||
466 | }; | ||
467 | |||
468 | enum osd_capability_bit_masks { | ||
469 | OSD_SEC_CAP_APPEND = BIT(0), | ||
470 | OSD_SEC_CAP_OBJ_MGMT = BIT(1), | ||
471 | OSD_SEC_CAP_REMOVE = BIT(2), | ||
472 | OSD_SEC_CAP_CREATE = BIT(3), | ||
473 | OSD_SEC_CAP_SET_ATTR = BIT(4), | ||
474 | OSD_SEC_CAP_GET_ATTR = BIT(5), | ||
475 | OSD_SEC_CAP_WRITE = BIT(6), | ||
476 | OSD_SEC_CAP_READ = BIT(7), | ||
477 | |||
478 | OSD_SEC_CAP_NONE1 = BIT(8), | ||
479 | OSD_SEC_CAP_NONE2 = BIT(9), | ||
480 | OSD_SEC_CAP_NONE3 = BIT(10), | ||
481 | OSD_SEC_CAP_QUERY = BIT(11), /*v2 only*/ | ||
482 | OSD_SEC_CAP_M_OBJECT = BIT(12), /*v2 only*/ | ||
483 | OSD_SEC_CAP_POL_SEC = BIT(13), | ||
484 | OSD_SEC_CAP_GLOBAL = BIT(14), | ||
485 | OSD_SEC_CAP_DEV_MGMT = BIT(15), | ||
486 | }; | ||
487 | |||
488 | /* for object_descriptor_type (hi nibble used) */ | ||
489 | enum { | ||
490 | OSD_SEC_OBJ_DESC_NONE = 0, /* Not allowed */ | ||
491 | OSD_SEC_OBJ_DESC_OBJ = 1 << 4, /* v1: also collection */ | ||
492 | OSD_SEC_OBJ_DESC_PAR = 2 << 4, /* also root */ | ||
493 | OSD_SEC_OBJ_DESC_COL = 3 << 4, /* v2 only */ | ||
494 | }; | ||
495 | |||
496 | /* (osd-r10:4.9.2.2) | ||
497 | * osd2r03:4.11.2.2 Capability format | ||
498 | */ | ||
499 | struct osd_capability_head { | ||
500 | u8 format; /* low nibble */ | ||
501 | u8 integrity_algorithm__key_version; /* MAKE_BYTE(integ_alg, key_ver) */ | ||
502 | u8 security_method; | ||
503 | u8 reserved1; | ||
504 | /*04*/ struct osd_timestamp expiration_time; | ||
505 | /*10*/ u8 audit[20]; | ||
506 | /*30*/ u8 discriminator[12]; | ||
507 | /*42*/ struct osd_timestamp object_created_time; | ||
508 | /*48*/ u8 object_type; | ||
509 | /*49*/ u8 permissions_bit_mask[5]; | ||
510 | /*54*/ u8 reserved2; | ||
511 | /*55*/ u8 object_descriptor_type; /* high nibble */ | ||
512 | } __packed; | ||
513 | |||
514 | /*56 v1*/ | ||
515 | struct osdv1_cap_object_descriptor { | ||
516 | union { | ||
517 | struct { | ||
518 | /*56*/ __be32 policy_access_tag; | ||
519 | /*60*/ __be64 allowed_partition_id; | ||
520 | /*68*/ __be64 allowed_object_id; | ||
521 | /*76*/ __be32 reserved; | ||
522 | } __packed obj_desc; | ||
523 | |||
524 | /*56*/ u8 object_descriptor[24]; | ||
525 | }; | ||
526 | } __packed; | ||
527 | /*80 v1*/ | ||
528 | |||
529 | /*56 v2*/ | ||
530 | struct osd_cap_object_descriptor { | ||
531 | union { | ||
532 | struct { | ||
533 | /*56*/ __be32 allowed_attributes_access; | ||
534 | /*60*/ __be32 policy_access_tag; | ||
535 | /*64*/ __be16 boot_epoch; | ||
536 | /*66*/ u8 reserved[6]; | ||
537 | /*72*/ __be64 allowed_partition_id; | ||
538 | /*80*/ __be64 allowed_object_id; | ||
539 | /*88*/ __be64 allowed_range_length; | ||
540 | /*96*/ __be64 allowed_range_start; | ||
541 | } __packed obj_desc; | ||
542 | |||
543 | /*56*/ u8 object_descriptor[48]; | ||
544 | }; | ||
545 | } __packed; | ||
546 | /*104 v2*/ | ||
547 | |||
548 | struct osdv1_capability { | ||
549 | struct osd_capability_head h; | ||
550 | struct osdv1_cap_object_descriptor od; | ||
551 | } __packed; | ||
552 | |||
553 | struct osd_capability { | ||
554 | struct osd_capability_head h; | ||
555 | /* struct osd_cap_object_descriptor od;*/ | ||
556 | struct osdv1_cap_object_descriptor od; /* FIXME: Pete rev-001 sup */ | ||
557 | } __packed; | ||
558 | |||
559 | /** | ||
560 | * osd_sec_set_caps - set cap-bits into the capabilities header | ||
561 | * | ||
562 | * @cap: The osd_capability_head to set cap bits to. | ||
563 | * @bit_mask: Use an ORed list of enum osd_capability_bit_masks values | ||
564 | * | ||
565 | * permissions_bit_mask is unaligned use below to set into caps | ||
566 | * in a version independent way | ||
567 | */ | ||
568 | static inline void osd_sec_set_caps(struct osd_capability_head *cap, | ||
569 | u16 bit_mask) | ||
570 | { | ||
571 | /* | ||
572 | *Note: The bits above are defined LE order this is because this way | ||
573 | * they can grow in the future to more then 16, and still retain | ||
574 | * there constant values. | ||
575 | */ | ||
576 | put_unaligned_le16(bit_mask, &cap->permissions_bit_mask); | ||
577 | } | ||
578 | |||
579 | #endif /* ndef __OSD_PROTOCOL_H__ */ | ||
diff --git a/include/scsi/osd_sec.h b/include/scsi/osd_sec.h new file mode 100644 index 000000000000..4c09fee8ae1e --- /dev/null +++ b/include/scsi/osd_sec.h | |||
@@ -0,0 +1,45 @@ | |||
1 | /* | ||
2 | * osd_sec.h - OSD security manager API | ||
3 | * | ||
4 | * Copyright (C) 2008 Panasas Inc. All rights reserved. | ||
5 | * | ||
6 | * Authors: | ||
7 | * Boaz Harrosh <bharrosh@panasas.com> | ||
8 | * Benny Halevy <bhalevy@panasas.com> | ||
9 | * | ||
10 | * This program is free software; you can redistribute it and/or modify | ||
11 | * it under the terms of the GNU General Public License version 2 | ||
12 | * | ||
13 | */ | ||
14 | #ifndef __OSD_SEC_H__ | ||
15 | #define __OSD_SEC_H__ | ||
16 | |||
17 | #include "osd_protocol.h" | ||
18 | #include "osd_types.h" | ||
19 | |||
20 | /* | ||
21 | * Contains types and constants of osd capabilities and security | ||
22 | * encoding/decoding. | ||
23 | * API is trying to keep security abstract so initiator of an object | ||
24 | * based pNFS client knows as little as possible about security and | ||
25 | * capabilities. It is the Server's osd-initiator place to know more. | ||
26 | * Also can be used by osd-target. | ||
27 | */ | ||
28 | void osd_sec_encode_caps(void *caps, ...);/* NI */ | ||
29 | void osd_sec_init_nosec_doall_caps(void *caps, | ||
30 | const struct osd_obj_id *obj, bool is_collection, const bool is_v1); | ||
31 | |||
32 | bool osd_is_sec_alldata(struct osd_security_parameters *sec_params); | ||
33 | |||
34 | /* Conditionally sign the CDB according to security setting in ocdb | ||
35 | * with cap_key */ | ||
36 | void osd_sec_sign_cdb(struct osd_cdb *ocdb, const u8 *cap_key); | ||
37 | |||
38 | /* Unconditionally sign the BIO data with cap_key. | ||
39 | * Check for osd_is_sec_alldata() was done prior to calling this. */ | ||
40 | void osd_sec_sign_data(void *data_integ, struct bio *bio, const u8 *cap_key); | ||
41 | |||
42 | /* Version independent copy of caps into the cdb */ | ||
43 | void osd_set_caps(struct osd_cdb *cdb, const void *caps); | ||
44 | |||
45 | #endif /* ndef __OSD_SEC_H__ */ | ||
diff --git a/include/scsi/osd_sense.h b/include/scsi/osd_sense.h new file mode 100644 index 000000000000..ff9b33c773c7 --- /dev/null +++ b/include/scsi/osd_sense.h | |||
@@ -0,0 +1,260 @@ | |||
1 | /* | ||
2 | * osd_sense.h - OSD Related sense handling definitions. | ||
3 | * | ||
4 | * Copyright (C) 2008 Panasas Inc. All rights reserved. | ||
5 | * | ||
6 | * Authors: | ||
7 | * Boaz Harrosh <bharrosh@panasas.com> | ||
8 | * Benny Halevy <bhalevy@panasas.com> | ||
9 | * | ||
10 | * This program is free software; you can redistribute it and/or modify | ||
11 | * it under the terms of the GNU General Public License version 2 | ||
12 | * | ||
13 | * This file contains types and constants that are defined by the protocol | ||
14 | * Note: All names and symbols are taken from the OSD standard's text. | ||
15 | */ | ||
16 | #ifndef __OSD_SENSE_H__ | ||
17 | #define __OSD_SENSE_H__ | ||
18 | |||
19 | #include <scsi/osd_protocol.h> | ||
20 | |||
21 | /* SPC3r23 4.5.6 Sense key and sense code definitions table 27 */ | ||
22 | enum scsi_sense_keys { | ||
23 | scsi_sk_no_sense = 0x0, | ||
24 | scsi_sk_recovered_error = 0x1, | ||
25 | scsi_sk_not_ready = 0x2, | ||
26 | scsi_sk_medium_error = 0x3, | ||
27 | scsi_sk_hardware_error = 0x4, | ||
28 | scsi_sk_illegal_request = 0x5, | ||
29 | scsi_sk_unit_attention = 0x6, | ||
30 | scsi_sk_data_protect = 0x7, | ||
31 | scsi_sk_blank_check = 0x8, | ||
32 | scsi_sk_vendor_specific = 0x9, | ||
33 | scsi_sk_copy_aborted = 0xa, | ||
34 | scsi_sk_aborted_command = 0xb, | ||
35 | scsi_sk_volume_overflow = 0xd, | ||
36 | scsi_sk_miscompare = 0xe, | ||
37 | scsi_sk_reserved = 0xf, | ||
38 | }; | ||
39 | |||
40 | /* SPC3r23 4.5.6 Sense key and sense code definitions table 28 */ | ||
41 | /* Note: only those which can be returned by an OSD target. Most of | ||
42 | * these errors are taken care of by the generic scsi layer. | ||
43 | */ | ||
44 | enum osd_additional_sense_codes { | ||
45 | scsi_no_additional_sense_information = 0x0000, | ||
46 | scsi_operation_in_progress = 0x0016, | ||
47 | scsi_cleaning_requested = 0x0017, | ||
48 | scsi_lunr_cause_not_reportable = 0x0400, | ||
49 | scsi_logical_unit_is_in_process_of_becoming_ready = 0x0401, | ||
50 | scsi_lunr_initializing_command_required = 0x0402, | ||
51 | scsi_lunr_manual_intervention_required = 0x0403, | ||
52 | scsi_lunr_operation_in_progress = 0x0407, | ||
53 | scsi_lunr_selftest_in_progress = 0x0409, | ||
54 | scsi_luna_asymmetric_access_state_transition = 0x040a, | ||
55 | scsi_luna_target_port_in_standby_state = 0x040b, | ||
56 | scsi_luna_target_port_in_unavailable_state = 0x040c, | ||
57 | scsi_lunr_notify_enable_spinup_required = 0x0411, | ||
58 | scsi_logical_unit_does_not_respond_to_selection = 0x0500, | ||
59 | scsi_logical_unit_communication_failure = 0x0800, | ||
60 | scsi_logical_unit_communication_timeout = 0x0801, | ||
61 | scsi_logical_unit_communication_parity_error = 0x0802, | ||
62 | scsi_error_log_overflow = 0x0a00, | ||
63 | scsi_warning = 0x0b00, | ||
64 | scsi_warning_specified_temperature_exceeded = 0x0b01, | ||
65 | scsi_warning_enclosure_degraded = 0x0b02, | ||
66 | scsi_write_error_unexpected_unsolicited_data = 0x0c0c, | ||
67 | scsi_write_error_not_enough_unsolicited_data = 0x0c0d, | ||
68 | scsi_invalid_information_unit = 0x0e00, | ||
69 | scsi_invalid_field_in_command_information_unit = 0x0e03, | ||
70 | scsi_read_error_failed_retransmission_request = 0x1113, | ||
71 | scsi_parameter_list_length_error = 0x1a00, | ||
72 | scsi_invalid_command_operation_code = 0x2000, | ||
73 | scsi_invalid_field_in_cdb = 0x2400, | ||
74 | osd_security_audit_value_frozen = 0x2404, | ||
75 | osd_security_working_key_frozen = 0x2405, | ||
76 | osd_nonce_not_unique = 0x2406, | ||
77 | osd_nonce_timestamp_out_of_range = 0x2407, | ||
78 | scsi_logical_unit_not_supported = 0x2500, | ||
79 | scsi_invalid_field_in_parameter_list = 0x2600, | ||
80 | scsi_parameter_not_supported = 0x2601, | ||
81 | scsi_parameter_value_invalid = 0x2602, | ||
82 | scsi_invalid_release_of_persistent_reservation = 0x2604, | ||
83 | osd_invalid_dataout_buffer_integrity_check_value = 0x260f, | ||
84 | scsi_not_ready_to_ready_change_medium_may_have_changed = 0x2800, | ||
85 | scsi_power_on_reset_or_bus_device_reset_occurred = 0x2900, | ||
86 | scsi_power_on_occurred = 0x2901, | ||
87 | scsi_scsi_bus_reset_occurred = 0x2902, | ||
88 | scsi_bus_device_reset_function_occurred = 0x2903, | ||
89 | scsi_device_internal_reset = 0x2904, | ||
90 | scsi_transceiver_mode_changed_to_single_ended = 0x2905, | ||
91 | scsi_transceiver_mode_changed_to_lvd = 0x2906, | ||
92 | scsi_i_t_nexus_loss_occurred = 0x2907, | ||
93 | scsi_parameters_changed = 0x2a00, | ||
94 | scsi_mode_parameters_changed = 0x2a01, | ||
95 | scsi_asymmetric_access_state_changed = 0x2a06, | ||
96 | scsi_priority_changed = 0x2a08, | ||
97 | scsi_command_sequence_error = 0x2c00, | ||
98 | scsi_previous_busy_status = 0x2c07, | ||
99 | scsi_previous_task_set_full_status = 0x2c08, | ||
100 | scsi_previous_reservation_conflict_status = 0x2c09, | ||
101 | osd_partition_or_collection_contains_user_objects = 0x2c0a, | ||
102 | scsi_commands_cleared_by_another_initiator = 0x2f00, | ||
103 | scsi_cleaning_failure = 0x3007, | ||
104 | scsi_enclosure_failure = 0x3400, | ||
105 | scsi_enclosure_services_failure = 0x3500, | ||
106 | scsi_unsupported_enclosure_function = 0x3501, | ||
107 | scsi_enclosure_services_unavailable = 0x3502, | ||
108 | scsi_enclosure_services_transfer_failure = 0x3503, | ||
109 | scsi_enclosure_services_transfer_refused = 0x3504, | ||
110 | scsi_enclosure_services_checksum_error = 0x3505, | ||
111 | scsi_rounded_parameter = 0x3700, | ||
112 | osd_read_past_end_of_user_object = 0x3b17, | ||
113 | scsi_logical_unit_has_not_self_configured_yet = 0x3e00, | ||
114 | scsi_logical_unit_failure = 0x3e01, | ||
115 | scsi_timeout_on_logical_unit = 0x3e02, | ||
116 | scsi_logical_unit_failed_selftest = 0x3e03, | ||
117 | scsi_logical_unit_unable_to_update_selftest_log = 0x3e04, | ||
118 | scsi_target_operating_conditions_have_changed = 0x3f00, | ||
119 | scsi_microcode_has_been_changed = 0x3f01, | ||
120 | scsi_inquiry_data_has_changed = 0x3f03, | ||
121 | scsi_echo_buffer_overwritten = 0x3f0f, | ||
122 | scsi_diagnostic_failure_on_component_nn_first = 0x4080, | ||
123 | scsi_diagnostic_failure_on_component_nn_last = 0x40ff, | ||
124 | scsi_message_error = 0x4300, | ||
125 | scsi_internal_target_failure = 0x4400, | ||
126 | scsi_select_or_reselect_failure = 0x4500, | ||
127 | scsi_scsi_parity_error = 0x4700, | ||
128 | scsi_data_phase_crc_error_detected = 0x4701, | ||
129 | scsi_scsi_parity_error_detected_during_st_data_phase = 0x4702, | ||
130 | scsi_asynchronous_information_protection_error_detected = 0x4704, | ||
131 | scsi_protocol_service_crc_error = 0x4705, | ||
132 | scsi_phy_test_function_in_progress = 0x4706, | ||
133 | scsi_invalid_message_error = 0x4900, | ||
134 | scsi_command_phase_error = 0x4a00, | ||
135 | scsi_data_phase_error = 0x4b00, | ||
136 | scsi_logical_unit_failed_self_configuration = 0x4c00, | ||
137 | scsi_overlapped_commands_attempted = 0x4e00, | ||
138 | osd_quota_error = 0x5507, | ||
139 | scsi_failure_prediction_threshold_exceeded = 0x5d00, | ||
140 | scsi_failure_prediction_threshold_exceeded_false = 0x5dff, | ||
141 | scsi_voltage_fault = 0x6500, | ||
142 | }; | ||
143 | |||
144 | enum scsi_descriptor_types { | ||
145 | scsi_sense_information = 0x0, | ||
146 | scsi_sense_command_specific_information = 0x1, | ||
147 | scsi_sense_key_specific = 0x2, | ||
148 | scsi_sense_field_replaceable_unit = 0x3, | ||
149 | scsi_sense_stream_commands = 0x4, | ||
150 | scsi_sense_block_commands = 0x5, | ||
151 | osd_sense_object_identification = 0x6, | ||
152 | osd_sense_response_integrity_check = 0x7, | ||
153 | osd_sense_attribute_identification = 0x8, | ||
154 | scsi_sense_ata_return = 0x9, | ||
155 | |||
156 | scsi_sense_Reserved_first = 0x0A, | ||
157 | scsi_sense_Reserved_last = 0x7F, | ||
158 | scsi_sense_Vendor_specific_first = 0x80, | ||
159 | scsi_sense_Vendor_specific_last = 0xFF, | ||
160 | }; | ||
161 | |||
162 | struct scsi_sense_descriptor { /* for picking into desc type */ | ||
163 | u8 descriptor_type; /* one of enum scsi_descriptor_types */ | ||
164 | u8 additional_length; /* n - 1 */ | ||
165 | u8 data[]; | ||
166 | } __packed; | ||
167 | |||
168 | /* OSD deploys only scsi descriptor_based sense buffers */ | ||
169 | struct scsi_sense_descriptor_based { | ||
170 | /*0*/ u8 response_code; /* 0x72 or 0x73 */ | ||
171 | /*1*/ u8 sense_key; /* one of enum scsi_sense_keys (4 lower bits) */ | ||
172 | /*2*/ __be16 additional_sense_code; /* enum osd_additional_sense_codes */ | ||
173 | /*4*/ u8 Reserved[3]; | ||
174 | /*7*/ u8 additional_sense_length; /* n - 7 */ | ||
175 | /*8*/ struct scsi_sense_descriptor ssd[0]; /* variable length, 1 or more */ | ||
176 | } __packed; | ||
177 | |||
178 | /* some descriptors deployed by OSD */ | ||
179 | |||
180 | /* SPC3r23 4.5.2.3 Command-specific information sense data descriptor */ | ||
181 | /* Note: this is the same for descriptor_type=00 but with type=00 the | ||
182 | * Reserved[0] == 0x80 (ie. bit-7 set) | ||
183 | */ | ||
184 | struct scsi_sense_command_specific_data_descriptor { | ||
185 | /*0*/ u8 descriptor_type; /* (00h/01h) */ | ||
186 | /*1*/ u8 additional_length; /* (0Ah) */ | ||
187 | /*2*/ u8 Reserved[2]; | ||
188 | /*4*/ __be64 information; | ||
189 | } __packed; | ||
190 | /*12*/ | ||
191 | |||
192 | struct scsi_sense_key_specific_data_descriptor { | ||
193 | /*0*/ u8 descriptor_type; /* (02h) */ | ||
194 | /*1*/ u8 additional_length; /* (06h) */ | ||
195 | /*2*/ u8 Reserved[2]; | ||
196 | /* SKSV, C/D, Reserved (2), BPV, BIT POINTER (3) */ | ||
197 | /*4*/ u8 sksv_cd_bpv_bp; | ||
198 | /*5*/ __be16 value; /* field-pointer/progress-value/retry-count/... */ | ||
199 | /*7*/ u8 Reserved2; | ||
200 | } __packed; | ||
201 | /*8*/ | ||
202 | |||
203 | /* 4.16.2.1 OSD error identification sense data descriptor - table 52 */ | ||
204 | /* Note: these bits are defined LE order for easy definition, this way the BIT() | ||
205 | * number is the same as in the documentation. Below members at | ||
206 | * osd_sense_identification_data_descriptor are therefore defined __le32. | ||
207 | */ | ||
208 | enum osd_command_functions_bits { | ||
209 | OSD_CFB_COMMAND = BIT(4), | ||
210 | OSD_CFB_CMD_CAP_VERIFIED = BIT(5), | ||
211 | OSD_CFB_VALIDATION = BIT(7), | ||
212 | OSD_CFB_IMP_ST_ATT = BIT(12), | ||
213 | OSD_CFB_SET_ATT = BIT(20), | ||
214 | OSD_CFB_SA_CAP_VERIFIED = BIT(21), | ||
215 | OSD_CFB_GET_ATT = BIT(28), | ||
216 | OSD_CFB_GA_CAP_VERIFIED = BIT(29), | ||
217 | }; | ||
218 | |||
219 | struct osd_sense_identification_data_descriptor { | ||
220 | /*0*/ u8 descriptor_type; /* (06h) */ | ||
221 | /*1*/ u8 additional_length; /* (1Eh) */ | ||
222 | /*2*/ u8 Reserved[6]; | ||
223 | /*8*/ __le32 not_initiated_functions; /*osd_command_functions_bits*/ | ||
224 | /*12*/ __le32 completed_functions; /*osd_command_functions_bits*/ | ||
225 | /*16*/ __be64 partition_id; | ||
226 | /*24*/ __be64 object_id; | ||
227 | } __packed; | ||
228 | /*32*/ | ||
229 | |||
230 | struct osd_sense_response_integrity_check_descriptor { | ||
231 | /*0*/ u8 descriptor_type; /* (07h) */ | ||
232 | /*1*/ u8 additional_length; /* (20h) */ | ||
233 | /*2*/ u8 integrity_check_value[32]; /*FIXME: OSDv2_CRYPTO_KEYID_SIZE*/ | ||
234 | } __packed; | ||
235 | /*34*/ | ||
236 | |||
237 | struct osd_sense_attributes_data_descriptor { | ||
238 | /*0*/ u8 descriptor_type; /* (08h) */ | ||
239 | /*1*/ u8 additional_length; /* (n-2) */ | ||
240 | /*2*/ u8 Reserved[6]; | ||
241 | struct osd_sense_attr { | ||
242 | /*8*/ __be32 attr_page; | ||
243 | /*12*/ __be32 attr_id; | ||
244 | /*16*/ } sense_attrs[0]; /* 1 or more */ | ||
245 | } __packed; | ||
246 | /*variable*/ | ||
247 | |||
248 | /* Dig into scsi_sk_illegal_request/scsi_invalid_field_in_cdb errors */ | ||
249 | |||
250 | /*FIXME: Support also field in CAPS*/ | ||
251 | #define OSD_CDB_OFFSET(F) offsetof(struct osd_cdb_head, F) | ||
252 | |||
253 | enum osdv2_cdb_field_offset { | ||
254 | OSDv1_CFO_STARTING_BYTE = OSD_CDB_OFFSET(v1.start_address), | ||
255 | OSD_CFO_STARTING_BYTE = OSD_CDB_OFFSET(v2.start_address), | ||
256 | OSD_CFO_PARTITION_ID = OSD_CDB_OFFSET(partition), | ||
257 | OSD_CFO_OBJECT_ID = OSD_CDB_OFFSET(object), | ||
258 | }; | ||
259 | |||
260 | #endif /* ndef __OSD_SENSE_H__ */ | ||
diff --git a/include/scsi/osd_types.h b/include/scsi/osd_types.h new file mode 100644 index 000000000000..3f5e88cc75c0 --- /dev/null +++ b/include/scsi/osd_types.h | |||
@@ -0,0 +1,40 @@ | |||
1 | /* | ||
2 | * osd_types.h - Types and constants which are not part of the protocol. | ||
3 | * | ||
4 | * Copyright (C) 2008 Panasas Inc. All rights reserved. | ||
5 | * | ||
6 | * Authors: | ||
7 | * Boaz Harrosh <bharrosh@panasas.com> | ||
8 | * Benny Halevy <bhalevy@panasas.com> | ||
9 | * | ||
10 | * This program is free software; you can redistribute it and/or modify | ||
11 | * it under the terms of the GNU General Public License version 2 | ||
12 | * | ||
13 | * Contains types and constants that are implementation specific and are | ||
14 | * used by more than one part of the osd library. | ||
15 | * (Eg initiator/target/security_manager/...) | ||
16 | */ | ||
17 | #ifndef __OSD_TYPES_H__ | ||
18 | #define __OSD_TYPES_H__ | ||
19 | |||
20 | struct osd_systemid { | ||
21 | u8 data[OSD_SYSTEMID_LEN]; | ||
22 | }; | ||
23 | |||
24 | typedef u64 __bitwise osd_id; | ||
25 | |||
26 | struct osd_obj_id { | ||
27 | osd_id partition; | ||
28 | osd_id id; | ||
29 | }; | ||
30 | |||
31 | static const struct __weak osd_obj_id osd_root_object = {0, 0}; | ||
32 | |||
33 | struct osd_attr { | ||
34 | u32 attr_page; | ||
35 | u32 attr_id; | ||
36 | u16 len; /* byte count of operand */ | ||
37 | void *val_ptr; /* in network order */ | ||
38 | }; | ||
39 | |||
40 | #endif /* ndef __OSD_TYPES_H__ */ | ||
diff --git a/include/scsi/scsi.h b/include/scsi/scsi.h index a109165714d6..084478e14d24 100644 --- a/include/scsi/scsi.h +++ b/include/scsi/scsi.h | |||
@@ -9,7 +9,8 @@ | |||
9 | #define _SCSI_SCSI_H | 9 | #define _SCSI_SCSI_H |
10 | 10 | ||
11 | #include <linux/types.h> | 11 | #include <linux/types.h> |
12 | #include <scsi/scsi_cmnd.h> | 12 | |
13 | struct scsi_cmnd; | ||
13 | 14 | ||
14 | /* | 15 | /* |
15 | * The maximum number of SG segments that we will put inside a | 16 | * The maximum number of SG segments that we will put inside a |
@@ -263,6 +264,7 @@ static inline int scsi_status_is_good(int status) | |||
263 | #define TYPE_RAID 0x0c | 264 | #define TYPE_RAID 0x0c |
264 | #define TYPE_ENCLOSURE 0x0d /* Enclosure Services Device */ | 265 | #define TYPE_ENCLOSURE 0x0d /* Enclosure Services Device */ |
265 | #define TYPE_RBC 0x0e | 266 | #define TYPE_RBC 0x0e |
267 | #define TYPE_OSD 0x11 | ||
266 | #define TYPE_NO_LUN 0x7f | 268 | #define TYPE_NO_LUN 0x7f |
267 | 269 | ||
268 | /* SCSI protocols; these are taken from SPC-3 section 7.5 */ | 270 | /* SCSI protocols; these are taken from SPC-3 section 7.5 */ |
@@ -402,16 +404,6 @@ static inline int scsi_is_wlun(unsigned int lun) | |||
402 | #define DRIVER_HARD 0x07 | 404 | #define DRIVER_HARD 0x07 |
403 | #define DRIVER_SENSE 0x08 | 405 | #define DRIVER_SENSE 0x08 |
404 | 406 | ||
405 | #define SUGGEST_RETRY 0x10 | ||
406 | #define SUGGEST_ABORT 0x20 | ||
407 | #define SUGGEST_REMAP 0x30 | ||
408 | #define SUGGEST_DIE 0x40 | ||
409 | #define SUGGEST_SENSE 0x80 | ||
410 | #define SUGGEST_IS_OK 0xff | ||
411 | |||
412 | #define DRIVER_MASK 0x0f | ||
413 | #define SUGGEST_MASK 0xf0 | ||
414 | |||
415 | /* | 407 | /* |
416 | * Internal return values. | 408 | * Internal return values. |
417 | */ | 409 | */ |
@@ -447,23 +439,6 @@ static inline int scsi_is_wlun(unsigned int lun) | |||
447 | #define msg_byte(result) (((result) >> 8) & 0xff) | 439 | #define msg_byte(result) (((result) >> 8) & 0xff) |
448 | #define host_byte(result) (((result) >> 16) & 0xff) | 440 | #define host_byte(result) (((result) >> 16) & 0xff) |
449 | #define driver_byte(result) (((result) >> 24) & 0xff) | 441 | #define driver_byte(result) (((result) >> 24) & 0xff) |
450 | #define suggestion(result) (driver_byte(result) & SUGGEST_MASK) | ||
451 | |||
452 | static inline void set_msg_byte(struct scsi_cmnd *cmd, char status) | ||
453 | { | ||
454 | cmd->result |= status << 8; | ||
455 | } | ||
456 | |||
457 | static inline void set_host_byte(struct scsi_cmnd *cmd, char status) | ||
458 | { | ||
459 | cmd->result |= status << 16; | ||
460 | } | ||
461 | |||
462 | static inline void set_driver_byte(struct scsi_cmnd *cmd, char status) | ||
463 | { | ||
464 | cmd->result |= status << 24; | ||
465 | } | ||
466 | |||
467 | 442 | ||
468 | #define sense_class(sense) (((sense) >> 4) & 0x7) | 443 | #define sense_class(sense) (((sense) >> 4) & 0x7) |
469 | #define sense_error(sense) ((sense) & 0xf) | 444 | #define sense_error(sense) ((sense) & 0xf) |
diff --git a/include/scsi/scsi_cmnd.h b/include/scsi/scsi_cmnd.h index 855bf95963e7..43b50d36925c 100644 --- a/include/scsi/scsi_cmnd.h +++ b/include/scsi/scsi_cmnd.h | |||
@@ -291,4 +291,19 @@ static inline struct scsi_data_buffer *scsi_prot(struct scsi_cmnd *cmd) | |||
291 | #define scsi_for_each_prot_sg(cmd, sg, nseg, __i) \ | 291 | #define scsi_for_each_prot_sg(cmd, sg, nseg, __i) \ |
292 | for_each_sg(scsi_prot_sglist(cmd), sg, nseg, __i) | 292 | for_each_sg(scsi_prot_sglist(cmd), sg, nseg, __i) |
293 | 293 | ||
294 | static inline void set_msg_byte(struct scsi_cmnd *cmd, char status) | ||
295 | { | ||
296 | cmd->result |= status << 8; | ||
297 | } | ||
298 | |||
299 | static inline void set_host_byte(struct scsi_cmnd *cmd, char status) | ||
300 | { | ||
301 | cmd->result |= status << 16; | ||
302 | } | ||
303 | |||
304 | static inline void set_driver_byte(struct scsi_cmnd *cmd, char status) | ||
305 | { | ||
306 | cmd->result |= status << 24; | ||
307 | } | ||
308 | |||
294 | #endif /* _SCSI_SCSI_CMND_H */ | 309 | #endif /* _SCSI_SCSI_CMND_H */ |
diff --git a/include/scsi/scsi_device.h b/include/scsi/scsi_device.h index 01a4c58f8bad..3f566af3f101 100644 --- a/include/scsi/scsi_device.h +++ b/include/scsi/scsi_device.h | |||
@@ -340,6 +340,7 @@ extern int scsi_mode_select(struct scsi_device *sdev, int pf, int sp, | |||
340 | struct scsi_sense_hdr *); | 340 | struct scsi_sense_hdr *); |
341 | extern int scsi_test_unit_ready(struct scsi_device *sdev, int timeout, | 341 | extern int scsi_test_unit_ready(struct scsi_device *sdev, int timeout, |
342 | int retries, struct scsi_sense_hdr *sshdr); | 342 | int retries, struct scsi_sense_hdr *sshdr); |
343 | extern unsigned char *scsi_get_vpd_page(struct scsi_device *, u8 page); | ||
343 | extern int scsi_device_set_state(struct scsi_device *sdev, | 344 | extern int scsi_device_set_state(struct scsi_device *sdev, |
344 | enum scsi_device_state state); | 345 | enum scsi_device_state state); |
345 | extern struct scsi_event *sdev_evt_alloc(enum scsi_device_event evt_type, | 346 | extern struct scsi_event *sdev_evt_alloc(enum scsi_device_event evt_type, |
@@ -370,12 +371,6 @@ extern int scsi_execute_req(struct scsi_device *sdev, const unsigned char *cmd, | |||
370 | int data_direction, void *buffer, unsigned bufflen, | 371 | int data_direction, void *buffer, unsigned bufflen, |
371 | struct scsi_sense_hdr *, int timeout, int retries, | 372 | struct scsi_sense_hdr *, int timeout, int retries, |
372 | int *resid); | 373 | int *resid); |
373 | extern int scsi_execute_async(struct scsi_device *sdev, | ||
374 | const unsigned char *cmd, int cmd_len, int data_direction, | ||
375 | void *buffer, unsigned bufflen, int use_sg, | ||
376 | int timeout, int retries, void *privdata, | ||
377 | void (*done)(void *, char *, int, int), | ||
378 | gfp_t gfp); | ||
379 | 374 | ||
380 | static inline int __must_check scsi_device_reprobe(struct scsi_device *sdev) | 375 | static inline int __must_check scsi_device_reprobe(struct scsi_device *sdev) |
381 | { | 376 | { |
@@ -400,7 +395,8 @@ static inline unsigned int sdev_id(struct scsi_device *sdev) | |||
400 | */ | 395 | */ |
401 | static inline int scsi_device_online(struct scsi_device *sdev) | 396 | static inline int scsi_device_online(struct scsi_device *sdev) |
402 | { | 397 | { |
403 | return sdev->sdev_state != SDEV_OFFLINE; | 398 | return (sdev->sdev_state != SDEV_OFFLINE && |
399 | sdev->sdev_state != SDEV_DEL); | ||
404 | } | 400 | } |
405 | static inline int scsi_device_blocked(struct scsi_device *sdev) | 401 | static inline int scsi_device_blocked(struct scsi_device *sdev) |
406 | { | 402 | { |
diff --git a/include/scsi/scsi_transport_iscsi.h b/include/scsi/scsi_transport_iscsi.h index b50aabe2861e..457588e1119b 100644 --- a/include/scsi/scsi_transport_iscsi.h +++ b/include/scsi/scsi_transport_iscsi.h | |||
@@ -88,7 +88,7 @@ struct iscsi_transport { | |||
88 | uint64_t host_param_mask; | 88 | uint64_t host_param_mask; |
89 | struct iscsi_cls_session *(*create_session) (struct iscsi_endpoint *ep, | 89 | struct iscsi_cls_session *(*create_session) (struct iscsi_endpoint *ep, |
90 | uint16_t cmds_max, uint16_t qdepth, | 90 | uint16_t cmds_max, uint16_t qdepth, |
91 | uint32_t sn, uint32_t *hn); | 91 | uint32_t sn); |
92 | void (*destroy_session) (struct iscsi_cls_session *session); | 92 | void (*destroy_session) (struct iscsi_cls_session *session); |
93 | struct iscsi_cls_conn *(*create_conn) (struct iscsi_cls_session *sess, | 93 | struct iscsi_cls_conn *(*create_conn) (struct iscsi_cls_session *sess, |
94 | uint32_t cid); | 94 | uint32_t cid); |
@@ -206,8 +206,6 @@ struct iscsi_cls_session { | |||
206 | struct iscsi_cls_host { | 206 | struct iscsi_cls_host { |
207 | atomic_t nr_scans; | 207 | atomic_t nr_scans; |
208 | struct mutex mutex; | 208 | struct mutex mutex; |
209 | struct workqueue_struct *scan_workq; | ||
210 | char scan_workq_name[20]; | ||
211 | }; | 209 | }; |
212 | 210 | ||
213 | extern void iscsi_host_for_each_session(struct Scsi_Host *shost, | 211 | extern void iscsi_host_for_each_session(struct Scsi_Host *shost, |