diff options
author | Glenn Elliott <gelliott@cs.unc.edu> | 2012-03-04 19:47:13 -0500 |
---|---|---|
committer | Glenn Elliott <gelliott@cs.unc.edu> | 2012-03-04 19:47:13 -0500 |
commit | c71c03bda1e86c9d5198c5d83f712e695c4f2a1e (patch) | |
tree | ecb166cb3e2b7e2adb3b5e292245fefd23381ac8 /drivers/s390/crypto | |
parent | ea53c912f8a86a8567697115b6a0d8152beee5c8 (diff) | |
parent | 6a00f206debf8a5c8899055726ad127dbeeed098 (diff) |
Merge branch 'mpi-master' into wip-k-fmlpwip-k-fmlp
Conflicts:
litmus/sched_cedf.c
Diffstat (limited to 'drivers/s390/crypto')
-rw-r--r-- | drivers/s390/crypto/ap_bus.c | 82 | ||||
-rw-r--r-- | drivers/s390/crypto/ap_bus.h | 2 | ||||
-rw-r--r-- | drivers/s390/crypto/zcrypt_api.c | 16 | ||||
-rw-r--r-- | drivers/s390/crypto/zcrypt_api.h | 3 | ||||
-rw-r--r-- | drivers/s390/crypto/zcrypt_cex2a.c | 82 | ||||
-rw-r--r-- | drivers/s390/crypto/zcrypt_cex2a.h | 25 | ||||
-rw-r--r-- | drivers/s390/crypto/zcrypt_pcica.c | 1 | ||||
-rw-r--r-- | drivers/s390/crypto/zcrypt_pcicc.c | 1 | ||||
-rw-r--r-- | drivers/s390/crypto/zcrypt_pcixcc.c | 17 |
9 files changed, 196 insertions, 33 deletions
diff --git a/drivers/s390/crypto/ap_bus.c b/drivers/s390/crypto/ap_bus.c index 91c6028d7b74..16e4a25596e7 100644 --- a/drivers/s390/crypto/ap_bus.c +++ b/drivers/s390/crypto/ap_bus.c | |||
@@ -27,6 +27,7 @@ | |||
27 | #define KMSG_COMPONENT "ap" | 27 | #define KMSG_COMPONENT "ap" |
28 | #define pr_fmt(fmt) KMSG_COMPONENT ": " fmt | 28 | #define pr_fmt(fmt) KMSG_COMPONENT ": " fmt |
29 | 29 | ||
30 | #include <linux/kernel_stat.h> | ||
30 | #include <linux/module.h> | 31 | #include <linux/module.h> |
31 | #include <linux/init.h> | 32 | #include <linux/init.h> |
32 | #include <linux/delay.h> | 33 | #include <linux/delay.h> |
@@ -154,14 +155,7 @@ static inline int ap_instructions_available(void) | |||
154 | */ | 155 | */ |
155 | static int ap_interrupts_available(void) | 156 | static int ap_interrupts_available(void) |
156 | { | 157 | { |
157 | unsigned long long facility_bits[2]; | 158 | return test_facility(2) && test_facility(65); |
158 | |||
159 | if (stfle(facility_bits, 2) <= 1) | ||
160 | return 0; | ||
161 | if (!(facility_bits[0] & (1ULL << 61)) || | ||
162 | !(facility_bits[1] & (1ULL << 62))) | ||
163 | return 0; | ||
164 | return 1; | ||
165 | } | 159 | } |
166 | 160 | ||
167 | /** | 161 | /** |
@@ -228,6 +222,69 @@ ap_queue_interruption_control(ap_qid_t qid, void *ind) | |||
228 | } | 222 | } |
229 | #endif | 223 | #endif |
230 | 224 | ||
225 | static inline struct ap_queue_status __ap_4096_commands_available(ap_qid_t qid, | ||
226 | int *support) | ||
227 | { | ||
228 | register unsigned long reg0 asm ("0") = 0UL | qid | (1UL << 23); | ||
229 | register struct ap_queue_status reg1 asm ("1"); | ||
230 | register unsigned long reg2 asm ("2") = 0UL; | ||
231 | |||
232 | asm volatile( | ||
233 | ".long 0xb2af0000\n" | ||
234 | "0: la %1,0\n" | ||
235 | "1:\n" | ||
236 | EX_TABLE(0b, 1b) | ||
237 | : "+d" (reg0), "=d" (reg1), "=d" (reg2) | ||
238 | : | ||
239 | : "cc"); | ||
240 | |||
241 | if (reg2 & 0x6000000000000000ULL) | ||
242 | *support = 1; | ||
243 | else | ||
244 | *support = 0; | ||
245 | |||
246 | return reg1; | ||
247 | } | ||
248 | |||
249 | /** | ||
250 | * ap_4096_commands_availablen(): Check for availability of 4096 bit RSA | ||
251 | * support. | ||
252 | * @qid: The AP queue number | ||
253 | * | ||
254 | * Returns 1 if 4096 bit RSA keys are support fo the AP, returns 0 if not. | ||
255 | */ | ||
256 | int ap_4096_commands_available(ap_qid_t qid) | ||
257 | { | ||
258 | struct ap_queue_status status; | ||
259 | int i, support = 0; | ||
260 | status = __ap_4096_commands_available(qid, &support); | ||
261 | |||
262 | for (i = 0; i < AP_MAX_RESET; i++) { | ||
263 | switch (status.response_code) { | ||
264 | case AP_RESPONSE_NORMAL: | ||
265 | return support; | ||
266 | case AP_RESPONSE_RESET_IN_PROGRESS: | ||
267 | case AP_RESPONSE_BUSY: | ||
268 | break; | ||
269 | case AP_RESPONSE_Q_NOT_AVAIL: | ||
270 | case AP_RESPONSE_DECONFIGURED: | ||
271 | case AP_RESPONSE_CHECKSTOPPED: | ||
272 | case AP_RESPONSE_INVALID_ADDRESS: | ||
273 | return 0; | ||
274 | case AP_RESPONSE_OTHERWISE_CHANGED: | ||
275 | break; | ||
276 | default: | ||
277 | break; | ||
278 | } | ||
279 | if (i < AP_MAX_RESET - 1) { | ||
280 | udelay(5); | ||
281 | status = __ap_4096_commands_available(qid, &support); | ||
282 | } | ||
283 | } | ||
284 | return support; | ||
285 | } | ||
286 | EXPORT_SYMBOL(ap_4096_commands_available); | ||
287 | |||
231 | /** | 288 | /** |
232 | * ap_queue_enable_interruption(): Enable interruption on an AP. | 289 | * ap_queue_enable_interruption(): Enable interruption on an AP. |
233 | * @qid: The AP queue number | 290 | * @qid: The AP queue number |
@@ -1049,6 +1106,7 @@ out: | |||
1049 | 1106 | ||
1050 | static void ap_interrupt_handler(void *unused1, void *unused2) | 1107 | static void ap_interrupt_handler(void *unused1, void *unused2) |
1051 | { | 1108 | { |
1109 | kstat_cpu(smp_processor_id()).irqs[IOINT_APB]++; | ||
1052 | tasklet_schedule(&ap_tasklet); | 1110 | tasklet_schedule(&ap_tasklet); |
1053 | } | 1111 | } |
1054 | 1112 | ||
@@ -1125,8 +1183,12 @@ static void ap_scan_bus(struct work_struct *unused) | |||
1125 | INIT_LIST_HEAD(&ap_dev->list); | 1183 | INIT_LIST_HEAD(&ap_dev->list); |
1126 | setup_timer(&ap_dev->timeout, ap_request_timeout, | 1184 | setup_timer(&ap_dev->timeout, ap_request_timeout, |
1127 | (unsigned long) ap_dev); | 1185 | (unsigned long) ap_dev); |
1128 | if (device_type == 0) | 1186 | if (device_type == 0) { |
1129 | ap_probe_device_type(ap_dev); | 1187 | if (ap_probe_device_type(ap_dev)) { |
1188 | kfree(ap_dev); | ||
1189 | continue; | ||
1190 | } | ||
1191 | } | ||
1130 | else | 1192 | else |
1131 | ap_dev->device_type = device_type; | 1193 | ap_dev->device_type = device_type; |
1132 | 1194 | ||
diff --git a/drivers/s390/crypto/ap_bus.h b/drivers/s390/crypto/ap_bus.h index 4785d07cd447..08b9738285b4 100644 --- a/drivers/s390/crypto/ap_bus.h +++ b/drivers/s390/crypto/ap_bus.h | |||
@@ -196,4 +196,6 @@ void ap_flush_queue(struct ap_device *ap_dev); | |||
196 | int ap_module_init(void); | 196 | int ap_module_init(void); |
197 | void ap_module_exit(void); | 197 | void ap_module_exit(void); |
198 | 198 | ||
199 | int ap_4096_commands_available(ap_qid_t qid); | ||
200 | |||
199 | #endif /* _AP_BUS_H_ */ | 201 | #endif /* _AP_BUS_H_ */ |
diff --git a/drivers/s390/crypto/zcrypt_api.c b/drivers/s390/crypto/zcrypt_api.c index 41e0aaefafd5..8e65447f76b7 100644 --- a/drivers/s390/crypto/zcrypt_api.c +++ b/drivers/s390/crypto/zcrypt_api.c | |||
@@ -35,7 +35,6 @@ | |||
35 | #include <linux/proc_fs.h> | 35 | #include <linux/proc_fs.h> |
36 | #include <linux/seq_file.h> | 36 | #include <linux/seq_file.h> |
37 | #include <linux/compat.h> | 37 | #include <linux/compat.h> |
38 | #include <linux/smp_lock.h> | ||
39 | #include <linux/slab.h> | 38 | #include <linux/slab.h> |
40 | #include <asm/atomic.h> | 39 | #include <asm/atomic.h> |
41 | #include <asm/uaccess.h> | 40 | #include <asm/uaccess.h> |
@@ -397,8 +396,15 @@ static long zcrypt_rsa_crt(struct ica_rsa_modexpo_crt *crt) | |||
397 | if (copied == 0) { | 396 | if (copied == 0) { |
398 | unsigned int len; | 397 | unsigned int len; |
399 | spin_unlock_bh(&zcrypt_device_lock); | 398 | spin_unlock_bh(&zcrypt_device_lock); |
400 | /* len is max 256 / 2 - 120 = 8 */ | 399 | /* len is max 256 / 2 - 120 = 8 |
401 | len = crt->inputdatalength / 2 - 120; | 400 | * For bigger device just assume len of leading |
401 | * 0s is 8 as stated in the requirements for | ||
402 | * ica_rsa_modexpo_crt struct in zcrypt.h. | ||
403 | */ | ||
404 | if (crt->inputdatalength <= 256) | ||
405 | len = crt->inputdatalength / 2 - 120; | ||
406 | else | ||
407 | len = 8; | ||
402 | if (len > sizeof(z1)) | 408 | if (len > sizeof(z1)) |
403 | return -EFAULT; | 409 | return -EFAULT; |
404 | z1 = z2 = z3 = 0; | 410 | z1 = z2 = z3 = 0; |
@@ -406,6 +412,7 @@ static long zcrypt_rsa_crt(struct ica_rsa_modexpo_crt *crt) | |||
406 | copy_from_user(&z2, crt->bp_key, len) || | 412 | copy_from_user(&z2, crt->bp_key, len) || |
407 | copy_from_user(&z3, crt->u_mult_inv, len)) | 413 | copy_from_user(&z3, crt->u_mult_inv, len)) |
408 | return -EFAULT; | 414 | return -EFAULT; |
415 | z1 = z2 = z3 = 0; | ||
409 | copied = 1; | 416 | copied = 1; |
410 | /* | 417 | /* |
411 | * We have to restart device lookup - | 418 | * We have to restart device lookup - |
@@ -897,7 +904,8 @@ static const struct file_operations zcrypt_fops = { | |||
897 | .compat_ioctl = zcrypt_compat_ioctl, | 904 | .compat_ioctl = zcrypt_compat_ioctl, |
898 | #endif | 905 | #endif |
899 | .open = zcrypt_open, | 906 | .open = zcrypt_open, |
900 | .release = zcrypt_release | 907 | .release = zcrypt_release, |
908 | .llseek = no_llseek, | ||
901 | }; | 909 | }; |
902 | 910 | ||
903 | /* | 911 | /* |
diff --git a/drivers/s390/crypto/zcrypt_api.h b/drivers/s390/crypto/zcrypt_api.h index 8e7ffbf2466c..9688f3985b07 100644 --- a/drivers/s390/crypto/zcrypt_api.h +++ b/drivers/s390/crypto/zcrypt_api.h | |||
@@ -76,7 +76,7 @@ struct ica_z90_status { | |||
76 | 76 | ||
77 | /** | 77 | /** |
78 | * Large random numbers are pulled in 4096 byte chunks from the crypto cards | 78 | * Large random numbers are pulled in 4096 byte chunks from the crypto cards |
79 | * and stored in a page. Be carefull when increasing this buffer due to size | 79 | * and stored in a page. Be careful when increasing this buffer due to size |
80 | * limitations for AP requests. | 80 | * limitations for AP requests. |
81 | */ | 81 | */ |
82 | #define ZCRYPT_RNG_BUFFER_SIZE 4096 | 82 | #define ZCRYPT_RNG_BUFFER_SIZE 4096 |
@@ -109,6 +109,7 @@ struct zcrypt_device { | |||
109 | int request_count; /* # current requests. */ | 109 | int request_count; /* # current requests. */ |
110 | 110 | ||
111 | struct ap_message reply; /* Per-device reply structure. */ | 111 | struct ap_message reply; /* Per-device reply structure. */ |
112 | int max_exp_bit_length; | ||
112 | }; | 113 | }; |
113 | 114 | ||
114 | struct zcrypt_device *zcrypt_device_alloc(size_t); | 115 | struct zcrypt_device *zcrypt_device_alloc(size_t); |
diff --git a/drivers/s390/crypto/zcrypt_cex2a.c b/drivers/s390/crypto/zcrypt_cex2a.c index 9c409efa1ecf..2176d00b395e 100644 --- a/drivers/s390/crypto/zcrypt_cex2a.c +++ b/drivers/s390/crypto/zcrypt_cex2a.c | |||
@@ -41,7 +41,7 @@ | |||
41 | #define CEX2A_MIN_MOD_SIZE 1 /* 8 bits */ | 41 | #define CEX2A_MIN_MOD_SIZE 1 /* 8 bits */ |
42 | #define CEX2A_MAX_MOD_SIZE 256 /* 2048 bits */ | 42 | #define CEX2A_MAX_MOD_SIZE 256 /* 2048 bits */ |
43 | #define CEX3A_MIN_MOD_SIZE CEX2A_MIN_MOD_SIZE | 43 | #define CEX3A_MIN_MOD_SIZE CEX2A_MIN_MOD_SIZE |
44 | #define CEX3A_MAX_MOD_SIZE CEX2A_MAX_MOD_SIZE | 44 | #define CEX3A_MAX_MOD_SIZE 512 /* 4096 bits */ |
45 | 45 | ||
46 | #define CEX2A_SPEED_RATING 970 | 46 | #define CEX2A_SPEED_RATING 970 |
47 | #define CEX3A_SPEED_RATING 900 /* Fixme: Needs finetuning */ | 47 | #define CEX3A_SPEED_RATING 900 /* Fixme: Needs finetuning */ |
@@ -49,8 +49,10 @@ | |||
49 | #define CEX2A_MAX_MESSAGE_SIZE 0x390 /* sizeof(struct type50_crb2_msg) */ | 49 | #define CEX2A_MAX_MESSAGE_SIZE 0x390 /* sizeof(struct type50_crb2_msg) */ |
50 | #define CEX2A_MAX_RESPONSE_SIZE 0x110 /* max outputdatalength + type80_hdr */ | 50 | #define CEX2A_MAX_RESPONSE_SIZE 0x110 /* max outputdatalength + type80_hdr */ |
51 | 51 | ||
52 | #define CEX3A_MAX_MESSAGE_SIZE CEX2A_MAX_MESSAGE_SIZE | 52 | #define CEX3A_MAX_RESPONSE_SIZE 0x210 /* 512 bit modulus |
53 | #define CEX3A_MAX_RESPONSE_SIZE CEX2A_MAX_RESPONSE_SIZE | 53 | * (max outputdatalength) + |
54 | * type80_hdr*/ | ||
55 | #define CEX3A_MAX_MESSAGE_SIZE sizeof(struct type50_crb3_msg) | ||
54 | 56 | ||
55 | #define CEX2A_CLEANUP_TIME (15*HZ) | 57 | #define CEX2A_CLEANUP_TIME (15*HZ) |
56 | #define CEX3A_CLEANUP_TIME CEX2A_CLEANUP_TIME | 58 | #define CEX3A_CLEANUP_TIME CEX2A_CLEANUP_TIME |
@@ -110,7 +112,7 @@ static int ICAMEX_msg_to_type50MEX_msg(struct zcrypt_device *zdev, | |||
110 | mod = meb1->modulus + sizeof(meb1->modulus) - mod_len; | 112 | mod = meb1->modulus + sizeof(meb1->modulus) - mod_len; |
111 | exp = meb1->exponent + sizeof(meb1->exponent) - mod_len; | 113 | exp = meb1->exponent + sizeof(meb1->exponent) - mod_len; |
112 | inp = meb1->message + sizeof(meb1->message) - mod_len; | 114 | inp = meb1->message + sizeof(meb1->message) - mod_len; |
113 | } else { | 115 | } else if (mod_len <= 256) { |
114 | struct type50_meb2_msg *meb2 = ap_msg->message; | 116 | struct type50_meb2_msg *meb2 = ap_msg->message; |
115 | memset(meb2, 0, sizeof(*meb2)); | 117 | memset(meb2, 0, sizeof(*meb2)); |
116 | ap_msg->length = sizeof(*meb2); | 118 | ap_msg->length = sizeof(*meb2); |
@@ -120,6 +122,17 @@ static int ICAMEX_msg_to_type50MEX_msg(struct zcrypt_device *zdev, | |||
120 | mod = meb2->modulus + sizeof(meb2->modulus) - mod_len; | 122 | mod = meb2->modulus + sizeof(meb2->modulus) - mod_len; |
121 | exp = meb2->exponent + sizeof(meb2->exponent) - mod_len; | 123 | exp = meb2->exponent + sizeof(meb2->exponent) - mod_len; |
122 | inp = meb2->message + sizeof(meb2->message) - mod_len; | 124 | inp = meb2->message + sizeof(meb2->message) - mod_len; |
125 | } else { | ||
126 | /* mod_len > 256 = 4096 bit RSA Key */ | ||
127 | struct type50_meb3_msg *meb3 = ap_msg->message; | ||
128 | memset(meb3, 0, sizeof(*meb3)); | ||
129 | ap_msg->length = sizeof(*meb3); | ||
130 | meb3->header.msg_type_code = TYPE50_TYPE_CODE; | ||
131 | meb3->header.msg_len = sizeof(*meb3); | ||
132 | meb3->keyblock_type = TYPE50_MEB3_FMT; | ||
133 | mod = meb3->modulus + sizeof(meb3->modulus) - mod_len; | ||
134 | exp = meb3->exponent + sizeof(meb3->exponent) - mod_len; | ||
135 | inp = meb3->message + sizeof(meb3->message) - mod_len; | ||
123 | } | 136 | } |
124 | 137 | ||
125 | if (copy_from_user(mod, mex->n_modulus, mod_len) || | 138 | if (copy_from_user(mod, mex->n_modulus, mod_len) || |
@@ -142,7 +155,7 @@ static int ICACRT_msg_to_type50CRT_msg(struct zcrypt_device *zdev, | |||
142 | struct ap_message *ap_msg, | 155 | struct ap_message *ap_msg, |
143 | struct ica_rsa_modexpo_crt *crt) | 156 | struct ica_rsa_modexpo_crt *crt) |
144 | { | 157 | { |
145 | int mod_len, short_len, long_len, long_offset; | 158 | int mod_len, short_len, long_len, long_offset, limit; |
146 | unsigned char *p, *q, *dp, *dq, *u, *inp; | 159 | unsigned char *p, *q, *dp, *dq, *u, *inp; |
147 | 160 | ||
148 | mod_len = crt->inputdatalength; | 161 | mod_len = crt->inputdatalength; |
@@ -152,14 +165,20 @@ static int ICACRT_msg_to_type50CRT_msg(struct zcrypt_device *zdev, | |||
152 | /* | 165 | /* |
153 | * CEX2A cannot handle p, dp, or U > 128 bytes. | 166 | * CEX2A cannot handle p, dp, or U > 128 bytes. |
154 | * If we have one of these, we need to do extra checking. | 167 | * If we have one of these, we need to do extra checking. |
168 | * For CEX3A the limit is 256 bytes. | ||
155 | */ | 169 | */ |
156 | if (long_len > 128) { | 170 | if (zdev->max_mod_size == CEX3A_MAX_MOD_SIZE) |
171 | limit = 256; | ||
172 | else | ||
173 | limit = 128; | ||
174 | |||
175 | if (long_len > limit) { | ||
157 | /* | 176 | /* |
158 | * zcrypt_rsa_crt already checked for the leading | 177 | * zcrypt_rsa_crt already checked for the leading |
159 | * zeroes of np_prime, bp_key and u_mult_inc. | 178 | * zeroes of np_prime, bp_key and u_mult_inc. |
160 | */ | 179 | */ |
161 | long_offset = long_len - 128; | 180 | long_offset = long_len - limit; |
162 | long_len = 128; | 181 | long_len = limit; |
163 | } else | 182 | } else |
164 | long_offset = 0; | 183 | long_offset = 0; |
165 | 184 | ||
@@ -180,7 +199,7 @@ static int ICACRT_msg_to_type50CRT_msg(struct zcrypt_device *zdev, | |||
180 | dq = crb1->dq + sizeof(crb1->dq) - short_len; | 199 | dq = crb1->dq + sizeof(crb1->dq) - short_len; |
181 | u = crb1->u + sizeof(crb1->u) - long_len; | 200 | u = crb1->u + sizeof(crb1->u) - long_len; |
182 | inp = crb1->message + sizeof(crb1->message) - mod_len; | 201 | inp = crb1->message + sizeof(crb1->message) - mod_len; |
183 | } else { | 202 | } else if (long_len <= 128) { |
184 | struct type50_crb2_msg *crb2 = ap_msg->message; | 203 | struct type50_crb2_msg *crb2 = ap_msg->message; |
185 | memset(crb2, 0, sizeof(*crb2)); | 204 | memset(crb2, 0, sizeof(*crb2)); |
186 | ap_msg->length = sizeof(*crb2); | 205 | ap_msg->length = sizeof(*crb2); |
@@ -193,6 +212,20 @@ static int ICACRT_msg_to_type50CRT_msg(struct zcrypt_device *zdev, | |||
193 | dq = crb2->dq + sizeof(crb2->dq) - short_len; | 212 | dq = crb2->dq + sizeof(crb2->dq) - short_len; |
194 | u = crb2->u + sizeof(crb2->u) - long_len; | 213 | u = crb2->u + sizeof(crb2->u) - long_len; |
195 | inp = crb2->message + sizeof(crb2->message) - mod_len; | 214 | inp = crb2->message + sizeof(crb2->message) - mod_len; |
215 | } else { | ||
216 | /* long_len >= 256 */ | ||
217 | struct type50_crb3_msg *crb3 = ap_msg->message; | ||
218 | memset(crb3, 0, sizeof(*crb3)); | ||
219 | ap_msg->length = sizeof(*crb3); | ||
220 | crb3->header.msg_type_code = TYPE50_TYPE_CODE; | ||
221 | crb3->header.msg_len = sizeof(*crb3); | ||
222 | crb3->keyblock_type = TYPE50_CRB3_FMT; | ||
223 | p = crb3->p + sizeof(crb3->p) - long_len; | ||
224 | q = crb3->q + sizeof(crb3->q) - short_len; | ||
225 | dp = crb3->dp + sizeof(crb3->dp) - long_len; | ||
226 | dq = crb3->dq + sizeof(crb3->dq) - short_len; | ||
227 | u = crb3->u + sizeof(crb3->u) - long_len; | ||
228 | inp = crb3->message + sizeof(crb3->message) - mod_len; | ||
196 | } | 229 | } |
197 | 230 | ||
198 | if (copy_from_user(p, crt->np_prime + long_offset, long_len) || | 231 | if (copy_from_user(p, crt->np_prime + long_offset, long_len) || |
@@ -203,7 +236,6 @@ static int ICACRT_msg_to_type50CRT_msg(struct zcrypt_device *zdev, | |||
203 | copy_from_user(inp, crt->inputdata, mod_len)) | 236 | copy_from_user(inp, crt->inputdata, mod_len)) |
204 | return -EFAULT; | 237 | return -EFAULT; |
205 | 238 | ||
206 | |||
207 | return 0; | 239 | return 0; |
208 | } | 240 | } |
209 | 241 | ||
@@ -230,7 +262,10 @@ static int convert_type80(struct zcrypt_device *zdev, | |||
230 | zdev->online = 0; | 262 | zdev->online = 0; |
231 | return -EAGAIN; /* repeat the request on a different device. */ | 263 | return -EAGAIN; /* repeat the request on a different device. */ |
232 | } | 264 | } |
233 | BUG_ON(t80h->len > CEX2A_MAX_RESPONSE_SIZE); | 265 | if (zdev->user_space_type == ZCRYPT_CEX2A) |
266 | BUG_ON(t80h->len > CEX2A_MAX_RESPONSE_SIZE); | ||
267 | else | ||
268 | BUG_ON(t80h->len > CEX3A_MAX_RESPONSE_SIZE); | ||
234 | data = reply->message + t80h->len - outputdatalength; | 269 | data = reply->message + t80h->len - outputdatalength; |
235 | if (copy_to_user(outputdata, data, outputdatalength)) | 270 | if (copy_to_user(outputdata, data, outputdatalength)) |
236 | return -EFAULT; | 271 | return -EFAULT; |
@@ -282,7 +317,10 @@ static void zcrypt_cex2a_receive(struct ap_device *ap_dev, | |||
282 | } | 317 | } |
283 | t80h = reply->message; | 318 | t80h = reply->message; |
284 | if (t80h->type == TYPE80_RSP_CODE) { | 319 | if (t80h->type == TYPE80_RSP_CODE) { |
285 | length = min(CEX2A_MAX_RESPONSE_SIZE, (int) t80h->len); | 320 | if (ap_dev->device_type == AP_DEVICE_TYPE_CEX2A) |
321 | length = min(CEX2A_MAX_RESPONSE_SIZE, (int) t80h->len); | ||
322 | else | ||
323 | length = min(CEX3A_MAX_RESPONSE_SIZE, (int) t80h->len); | ||
286 | memcpy(msg->message, reply->message, length); | 324 | memcpy(msg->message, reply->message, length); |
287 | } else | 325 | } else |
288 | memcpy(msg->message, reply->message, sizeof error_reply); | 326 | memcpy(msg->message, reply->message, sizeof error_reply); |
@@ -307,7 +345,10 @@ static long zcrypt_cex2a_modexpo(struct zcrypt_device *zdev, | |||
307 | int rc; | 345 | int rc; |
308 | 346 | ||
309 | ap_init_message(&ap_msg); | 347 | ap_init_message(&ap_msg); |
310 | ap_msg.message = kmalloc(CEX2A_MAX_MESSAGE_SIZE, GFP_KERNEL); | 348 | if (zdev->user_space_type == ZCRYPT_CEX2A) |
349 | ap_msg.message = kmalloc(CEX2A_MAX_MESSAGE_SIZE, GFP_KERNEL); | ||
350 | else | ||
351 | ap_msg.message = kmalloc(CEX3A_MAX_MESSAGE_SIZE, GFP_KERNEL); | ||
311 | if (!ap_msg.message) | 352 | if (!ap_msg.message) |
312 | return -ENOMEM; | 353 | return -ENOMEM; |
313 | ap_msg.psmid = (((unsigned long long) current->pid) << 32) + | 354 | ap_msg.psmid = (((unsigned long long) current->pid) << 32) + |
@@ -345,7 +386,10 @@ static long zcrypt_cex2a_modexpo_crt(struct zcrypt_device *zdev, | |||
345 | int rc; | 386 | int rc; |
346 | 387 | ||
347 | ap_init_message(&ap_msg); | 388 | ap_init_message(&ap_msg); |
348 | ap_msg.message = kmalloc(CEX2A_MAX_MESSAGE_SIZE, GFP_KERNEL); | 389 | if (zdev->user_space_type == ZCRYPT_CEX2A) |
390 | ap_msg.message = kmalloc(CEX2A_MAX_MESSAGE_SIZE, GFP_KERNEL); | ||
391 | else | ||
392 | ap_msg.message = kmalloc(CEX3A_MAX_MESSAGE_SIZE, GFP_KERNEL); | ||
349 | if (!ap_msg.message) | 393 | if (!ap_msg.message) |
350 | return -ENOMEM; | 394 | return -ENOMEM; |
351 | ap_msg.psmid = (((unsigned long long) current->pid) << 32) + | 395 | ap_msg.psmid = (((unsigned long long) current->pid) << 32) + |
@@ -397,6 +441,7 @@ static int zcrypt_cex2a_probe(struct ap_device *ap_dev) | |||
397 | zdev->max_mod_size = CEX2A_MAX_MOD_SIZE; | 441 | zdev->max_mod_size = CEX2A_MAX_MOD_SIZE; |
398 | zdev->short_crt = 1; | 442 | zdev->short_crt = 1; |
399 | zdev->speed_rating = CEX2A_SPEED_RATING; | 443 | zdev->speed_rating = CEX2A_SPEED_RATING; |
444 | zdev->max_exp_bit_length = CEX2A_MAX_MOD_SIZE; | ||
400 | break; | 445 | break; |
401 | case AP_DEVICE_TYPE_CEX3A: | 446 | case AP_DEVICE_TYPE_CEX3A: |
402 | zdev = zcrypt_device_alloc(CEX3A_MAX_RESPONSE_SIZE); | 447 | zdev = zcrypt_device_alloc(CEX3A_MAX_RESPONSE_SIZE); |
@@ -404,8 +449,13 @@ static int zcrypt_cex2a_probe(struct ap_device *ap_dev) | |||
404 | return -ENOMEM; | 449 | return -ENOMEM; |
405 | zdev->user_space_type = ZCRYPT_CEX3A; | 450 | zdev->user_space_type = ZCRYPT_CEX3A; |
406 | zdev->type_string = "CEX3A"; | 451 | zdev->type_string = "CEX3A"; |
407 | zdev->min_mod_size = CEX3A_MIN_MOD_SIZE; | 452 | zdev->min_mod_size = CEX2A_MIN_MOD_SIZE; |
408 | zdev->max_mod_size = CEX3A_MAX_MOD_SIZE; | 453 | zdev->max_mod_size = CEX2A_MAX_MOD_SIZE; |
454 | zdev->max_exp_bit_length = CEX2A_MAX_MOD_SIZE; | ||
455 | if (ap_4096_commands_available(ap_dev->qid)) { | ||
456 | zdev->max_mod_size = CEX3A_MAX_MOD_SIZE; | ||
457 | zdev->max_exp_bit_length = CEX3A_MAX_MOD_SIZE; | ||
458 | } | ||
409 | zdev->short_crt = 1; | 459 | zdev->short_crt = 1; |
410 | zdev->speed_rating = CEX3A_SPEED_RATING; | 460 | zdev->speed_rating = CEX3A_SPEED_RATING; |
411 | break; | 461 | break; |
diff --git a/drivers/s390/crypto/zcrypt_cex2a.h b/drivers/s390/crypto/zcrypt_cex2a.h index 8f69d1dacab8..0350665810cf 100644 --- a/drivers/s390/crypto/zcrypt_cex2a.h +++ b/drivers/s390/crypto/zcrypt_cex2a.h | |||
@@ -51,8 +51,10 @@ struct type50_hdr { | |||
51 | 51 | ||
52 | #define TYPE50_MEB1_FMT 0x0001 | 52 | #define TYPE50_MEB1_FMT 0x0001 |
53 | #define TYPE50_MEB2_FMT 0x0002 | 53 | #define TYPE50_MEB2_FMT 0x0002 |
54 | #define TYPE50_MEB3_FMT 0x0003 | ||
54 | #define TYPE50_CRB1_FMT 0x0011 | 55 | #define TYPE50_CRB1_FMT 0x0011 |
55 | #define TYPE50_CRB2_FMT 0x0012 | 56 | #define TYPE50_CRB2_FMT 0x0012 |
57 | #define TYPE50_CRB3_FMT 0x0013 | ||
56 | 58 | ||
57 | /* Mod-Exp, with a small modulus */ | 59 | /* Mod-Exp, with a small modulus */ |
58 | struct type50_meb1_msg { | 60 | struct type50_meb1_msg { |
@@ -74,6 +76,16 @@ struct type50_meb2_msg { | |||
74 | unsigned char message[256]; | 76 | unsigned char message[256]; |
75 | } __attribute__((packed)); | 77 | } __attribute__((packed)); |
76 | 78 | ||
79 | /* Mod-Exp, with a larger modulus */ | ||
80 | struct type50_meb3_msg { | ||
81 | struct type50_hdr header; | ||
82 | unsigned short keyblock_type; /* 0x0003 */ | ||
83 | unsigned char reserved[6]; | ||
84 | unsigned char exponent[512]; | ||
85 | unsigned char modulus[512]; | ||
86 | unsigned char message[512]; | ||
87 | } __attribute__((packed)); | ||
88 | |||
77 | /* CRT, with a small modulus */ | 89 | /* CRT, with a small modulus */ |
78 | struct type50_crb1_msg { | 90 | struct type50_crb1_msg { |
79 | struct type50_hdr header; | 91 | struct type50_hdr header; |
@@ -100,6 +112,19 @@ struct type50_crb2_msg { | |||
100 | unsigned char message[256]; | 112 | unsigned char message[256]; |
101 | } __attribute__((packed)); | 113 | } __attribute__((packed)); |
102 | 114 | ||
115 | /* CRT, with a larger modulus */ | ||
116 | struct type50_crb3_msg { | ||
117 | struct type50_hdr header; | ||
118 | unsigned short keyblock_type; /* 0x0013 */ | ||
119 | unsigned char reserved[6]; | ||
120 | unsigned char p[256]; | ||
121 | unsigned char q[256]; | ||
122 | unsigned char dp[256]; | ||
123 | unsigned char dq[256]; | ||
124 | unsigned char u[256]; | ||
125 | unsigned char message[512]; | ||
126 | } __attribute__((packed)); | ||
127 | |||
103 | /** | 128 | /** |
104 | * The type 80 response family is associated with a CEX2A card. | 129 | * The type 80 response family is associated with a CEX2A card. |
105 | * | 130 | * |
diff --git a/drivers/s390/crypto/zcrypt_pcica.c b/drivers/s390/crypto/zcrypt_pcica.c index 09e934b295a0..1afb69c75fea 100644 --- a/drivers/s390/crypto/zcrypt_pcica.c +++ b/drivers/s390/crypto/zcrypt_pcica.c | |||
@@ -373,6 +373,7 @@ static int zcrypt_pcica_probe(struct ap_device *ap_dev) | |||
373 | zdev->min_mod_size = PCICA_MIN_MOD_SIZE; | 373 | zdev->min_mod_size = PCICA_MIN_MOD_SIZE; |
374 | zdev->max_mod_size = PCICA_MAX_MOD_SIZE; | 374 | zdev->max_mod_size = PCICA_MAX_MOD_SIZE; |
375 | zdev->speed_rating = PCICA_SPEED_RATING; | 375 | zdev->speed_rating = PCICA_SPEED_RATING; |
376 | zdev->max_exp_bit_length = PCICA_MAX_MOD_SIZE; | ||
376 | ap_dev->reply = &zdev->reply; | 377 | ap_dev->reply = &zdev->reply; |
377 | ap_dev->private = zdev; | 378 | ap_dev->private = zdev; |
378 | rc = zcrypt_device_register(zdev); | 379 | rc = zcrypt_device_register(zdev); |
diff --git a/drivers/s390/crypto/zcrypt_pcicc.c b/drivers/s390/crypto/zcrypt_pcicc.c index 9dec5c77cff4..aa4c050a5694 100644 --- a/drivers/s390/crypto/zcrypt_pcicc.c +++ b/drivers/s390/crypto/zcrypt_pcicc.c | |||
@@ -579,6 +579,7 @@ static int zcrypt_pcicc_probe(struct ap_device *ap_dev) | |||
579 | zdev->min_mod_size = PCICC_MIN_MOD_SIZE; | 579 | zdev->min_mod_size = PCICC_MIN_MOD_SIZE; |
580 | zdev->max_mod_size = PCICC_MAX_MOD_SIZE; | 580 | zdev->max_mod_size = PCICC_MAX_MOD_SIZE; |
581 | zdev->speed_rating = PCICC_SPEED_RATING; | 581 | zdev->speed_rating = PCICC_SPEED_RATING; |
582 | zdev->max_exp_bit_length = PCICC_MAX_MOD_SIZE; | ||
582 | ap_dev->reply = &zdev->reply; | 583 | ap_dev->reply = &zdev->reply; |
583 | ap_dev->private = zdev; | 584 | ap_dev->private = zdev; |
584 | rc = zcrypt_device_register(zdev); | 585 | rc = zcrypt_device_register(zdev); |
diff --git a/drivers/s390/crypto/zcrypt_pcixcc.c b/drivers/s390/crypto/zcrypt_pcixcc.c index 510fab4577d4..4f85eb725f4f 100644 --- a/drivers/s390/crypto/zcrypt_pcixcc.c +++ b/drivers/s390/crypto/zcrypt_pcixcc.c | |||
@@ -45,12 +45,12 @@ | |||
45 | #define PCIXCC_MIN_MOD_SIZE_OLD 64 /* 512 bits */ | 45 | #define PCIXCC_MIN_MOD_SIZE_OLD 64 /* 512 bits */ |
46 | #define PCIXCC_MAX_MOD_SIZE 256 /* 2048 bits */ | 46 | #define PCIXCC_MAX_MOD_SIZE 256 /* 2048 bits */ |
47 | #define CEX3C_MIN_MOD_SIZE PCIXCC_MIN_MOD_SIZE | 47 | #define CEX3C_MIN_MOD_SIZE PCIXCC_MIN_MOD_SIZE |
48 | #define CEX3C_MAX_MOD_SIZE PCIXCC_MAX_MOD_SIZE | 48 | #define CEX3C_MAX_MOD_SIZE 512 /* 4096 bits */ |
49 | 49 | ||
50 | #define PCIXCC_MCL2_SPEED_RATING 7870 | 50 | #define PCIXCC_MCL2_SPEED_RATING 7870 |
51 | #define PCIXCC_MCL3_SPEED_RATING 7870 | 51 | #define PCIXCC_MCL3_SPEED_RATING 7870 |
52 | #define CEX2C_SPEED_RATING 7000 | 52 | #define CEX2C_SPEED_RATING 7000 |
53 | #define CEX3C_SPEED_RATING 6500 /* FIXME: needs finetuning */ | 53 | #define CEX3C_SPEED_RATING 6500 |
54 | 54 | ||
55 | #define PCIXCC_MAX_ICA_MESSAGE_SIZE 0x77c /* max size type6 v2 crt message */ | 55 | #define PCIXCC_MAX_ICA_MESSAGE_SIZE 0x77c /* max size type6 v2 crt message */ |
56 | #define PCIXCC_MAX_ICA_RESPONSE_SIZE 0x77c /* max size type86 v2 reply */ | 56 | #define PCIXCC_MAX_ICA_RESPONSE_SIZE 0x77c /* max size type86 v2 reply */ |
@@ -567,6 +567,15 @@ static int convert_response_ica(struct zcrypt_device *zdev, | |||
567 | case TYPE88_RSP_CODE: | 567 | case TYPE88_RSP_CODE: |
568 | return convert_error(zdev, reply); | 568 | return convert_error(zdev, reply); |
569 | case TYPE86_RSP_CODE: | 569 | case TYPE86_RSP_CODE: |
570 | if (msg->cprbx.ccp_rtcode && | ||
571 | (msg->cprbx.ccp_rscode == 0x14f) && | ||
572 | (outputdatalength > 256)) { | ||
573 | if (zdev->max_exp_bit_length <= 17) { | ||
574 | zdev->max_exp_bit_length = 17; | ||
575 | return -EAGAIN; | ||
576 | } else | ||
577 | return -EINVAL; | ||
578 | } | ||
570 | if (msg->hdr.reply_code) | 579 | if (msg->hdr.reply_code) |
571 | return convert_error(zdev, reply); | 580 | return convert_error(zdev, reply); |
572 | if (msg->cprbx.cprb_ver_id == 0x02) | 581 | if (msg->cprbx.cprb_ver_id == 0x02) |
@@ -1052,11 +1061,13 @@ static int zcrypt_pcixcc_probe(struct ap_device *ap_dev) | |||
1052 | zdev->speed_rating = PCIXCC_MCL2_SPEED_RATING; | 1061 | zdev->speed_rating = PCIXCC_MCL2_SPEED_RATING; |
1053 | zdev->min_mod_size = PCIXCC_MIN_MOD_SIZE_OLD; | 1062 | zdev->min_mod_size = PCIXCC_MIN_MOD_SIZE_OLD; |
1054 | zdev->max_mod_size = PCIXCC_MAX_MOD_SIZE; | 1063 | zdev->max_mod_size = PCIXCC_MAX_MOD_SIZE; |
1064 | zdev->max_exp_bit_length = PCIXCC_MAX_MOD_SIZE; | ||
1055 | } else { | 1065 | } else { |
1056 | zdev->type_string = "PCIXCC_MCL3"; | 1066 | zdev->type_string = "PCIXCC_MCL3"; |
1057 | zdev->speed_rating = PCIXCC_MCL3_SPEED_RATING; | 1067 | zdev->speed_rating = PCIXCC_MCL3_SPEED_RATING; |
1058 | zdev->min_mod_size = PCIXCC_MIN_MOD_SIZE; | 1068 | zdev->min_mod_size = PCIXCC_MIN_MOD_SIZE; |
1059 | zdev->max_mod_size = PCIXCC_MAX_MOD_SIZE; | 1069 | zdev->max_mod_size = PCIXCC_MAX_MOD_SIZE; |
1070 | zdev->max_exp_bit_length = PCIXCC_MAX_MOD_SIZE; | ||
1060 | } | 1071 | } |
1061 | break; | 1072 | break; |
1062 | case AP_DEVICE_TYPE_CEX2C: | 1073 | case AP_DEVICE_TYPE_CEX2C: |
@@ -1065,6 +1076,7 @@ static int zcrypt_pcixcc_probe(struct ap_device *ap_dev) | |||
1065 | zdev->speed_rating = CEX2C_SPEED_RATING; | 1076 | zdev->speed_rating = CEX2C_SPEED_RATING; |
1066 | zdev->min_mod_size = PCIXCC_MIN_MOD_SIZE; | 1077 | zdev->min_mod_size = PCIXCC_MIN_MOD_SIZE; |
1067 | zdev->max_mod_size = PCIXCC_MAX_MOD_SIZE; | 1078 | zdev->max_mod_size = PCIXCC_MAX_MOD_SIZE; |
1079 | zdev->max_exp_bit_length = PCIXCC_MAX_MOD_SIZE; | ||
1068 | break; | 1080 | break; |
1069 | case AP_DEVICE_TYPE_CEX3C: | 1081 | case AP_DEVICE_TYPE_CEX3C: |
1070 | zdev->user_space_type = ZCRYPT_CEX3C; | 1082 | zdev->user_space_type = ZCRYPT_CEX3C; |
@@ -1072,6 +1084,7 @@ static int zcrypt_pcixcc_probe(struct ap_device *ap_dev) | |||
1072 | zdev->speed_rating = CEX3C_SPEED_RATING; | 1084 | zdev->speed_rating = CEX3C_SPEED_RATING; |
1073 | zdev->min_mod_size = CEX3C_MIN_MOD_SIZE; | 1085 | zdev->min_mod_size = CEX3C_MIN_MOD_SIZE; |
1074 | zdev->max_mod_size = CEX3C_MAX_MOD_SIZE; | 1086 | zdev->max_mod_size = CEX3C_MAX_MOD_SIZE; |
1087 | zdev->max_exp_bit_length = CEX3C_MAX_MOD_SIZE; | ||
1075 | break; | 1088 | break; |
1076 | default: | 1089 | default: |
1077 | goto out_free; | 1090 | goto out_free; |