aboutsummaryrefslogtreecommitdiffstats
path: root/drivers
diff options
context:
space:
mode:
authorLinus Torvalds <torvalds@linux-foundation.org>2015-02-11 20:42:32 -0500
committerLinus Torvalds <torvalds@linux-foundation.org>2015-02-11 20:42:32 -0500
commitb3d6524ff7956c5a898d51a18eaecb62a60a2b84 (patch)
treecc049e7ec9edd9f5a76f286e04d8db9a1caa516a /drivers
parent07f80d41cf24b7e6e76cd97d420167932c9a7f82 (diff)
parent6a039eab53c01a58bfff95c78fc800ca7de27c77 (diff)
Merge branch 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/s390/linux
Pull s390 updates from Martin Schwidefsky: - The remaining patches for the z13 machine support: kernel build option for z13, the cache synonym avoidance, SMT support, compare-and-delay for spinloops and the CES5S crypto adapater. - The ftrace support for function tracing with the gcc hotpatch option. This touches common code Makefiles, Steven is ok with the changes. - The hypfs file system gets an extension to access diagnose 0x0c data in user space for performance analysis for Linux running under z/VM. - The iucv hvc console gets wildcard spport for the user id filtering. - The cacheinfo code is converted to use the generic infrastructure. - Cleanup and bug fixes. * 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/s390/linux: (42 commits) s390/process: free vx save area when releasing tasks s390/hypfs: Eliminate hypfs interval s390/hypfs: Add diagnose 0c support s390/cacheinfo: don't use smp_processor_id() in preemptible context s390/zcrypt: fixed domain scanning problem (again) s390/smp: increase maximum value of NR_CPUS to 512 s390/jump label: use different nop instruction s390/jump label: add sanity checks s390/mm: correct missing space when reporting user process faults s390/dasd: cleanup profiling s390/dasd: add locking for global_profile access s390/ftrace: hotpatch support for function tracing ftrace: let notrace function attribute disable hotpatching if necessary ftrace: allow architectures to specify ftrace compile options s390: reintroduce diag 44 calls for cpu_relax() s390/zcrypt: Add support for new crypto express (CEX5S) adapter. s390/zcrypt: Number of supported ap domains is not retrievable. s390/spinlock: add compare-and-delay to lock wait loops s390/tape: remove redundant if statement s390/hvc_iucv: add simple wildcard matches to the iucv allow filter ...
Diffstat (limited to 'drivers')
-rw-r--r--drivers/s390/block/dasd.c102
-rw-r--r--drivers/s390/block/dasd_int.h3
-rw-r--r--drivers/s390/block/dasd_proc.c21
-rw-r--r--drivers/s390/block/dcssblk.c15
-rw-r--r--drivers/s390/char/hmcdrv_ftp.c6
-rw-r--r--drivers/s390/char/hmcdrv_mod.c1
-rw-r--r--drivers/s390/char/sclp_early.c49
-rw-r--r--drivers/s390/char/tape_34xx.c12
-rw-r--r--drivers/s390/cio/cio.c2
-rw-r--r--drivers/s390/cio/idset.c20
-rw-r--r--drivers/s390/cio/idset.h2
-rw-r--r--drivers/s390/crypto/ap_bus.c144
-rw-r--r--drivers/s390/crypto/ap_bus.h1
-rw-r--r--drivers/s390/crypto/zcrypt_api.h1
-rw-r--r--drivers/s390/crypto/zcrypt_cex4.c33
-rw-r--r--drivers/tty/hvc/hvc_iucv.c31
16 files changed, 253 insertions, 190 deletions
diff --git a/drivers/s390/block/dasd.c b/drivers/s390/block/dasd.c
index 4abf11965484..be34ef41b7c7 100644
--- a/drivers/s390/block/dasd.c
+++ b/drivers/s390/block/dasd.c
@@ -674,8 +674,9 @@ EXPORT_SYMBOL(dasd_enable_device);
674unsigned int dasd_global_profile_level = DASD_PROFILE_OFF; 674unsigned int dasd_global_profile_level = DASD_PROFILE_OFF;
675 675
676#ifdef CONFIG_DASD_PROFILE 676#ifdef CONFIG_DASD_PROFILE
677struct dasd_profile_info dasd_global_profile_data; 677struct dasd_profile dasd_global_profile = {
678static struct dentry *dasd_global_profile_dentry; 678 .lock = __SPIN_LOCK_UNLOCKED(dasd_global_profile.lock),
679};
679static struct dentry *dasd_debugfs_global_entry; 680static struct dentry *dasd_debugfs_global_entry;
680 681
681/* 682/*
@@ -696,11 +697,13 @@ static void dasd_profile_start(struct dasd_block *block,
696 if (++counter >= 31) 697 if (++counter >= 31)
697 break; 698 break;
698 699
699 if (dasd_global_profile_level) { 700 spin_lock(&dasd_global_profile.lock);
700 dasd_global_profile_data.dasd_io_nr_req[counter]++; 701 if (dasd_global_profile.data) {
702 dasd_global_profile.data->dasd_io_nr_req[counter]++;
701 if (rq_data_dir(req) == READ) 703 if (rq_data_dir(req) == READ)
702 dasd_global_profile_data.dasd_read_nr_req[counter]++; 704 dasd_global_profile.data->dasd_read_nr_req[counter]++;
703 } 705 }
706 spin_unlock(&dasd_global_profile.lock);
704 707
705 spin_lock(&block->profile.lock); 708 spin_lock(&block->profile.lock);
706 if (block->profile.data) { 709 if (block->profile.data) {
@@ -825,8 +828,9 @@ static void dasd_profile_end(struct dasd_block *block,
825 dasd_profile_counter(irqtime / sectors, irqtimeps_ind); 828 dasd_profile_counter(irqtime / sectors, irqtimeps_ind);
826 dasd_profile_counter(endtime, endtime_ind); 829 dasd_profile_counter(endtime, endtime_ind);
827 830
828 if (dasd_global_profile_level) { 831 spin_lock(&dasd_global_profile.lock);
829 dasd_profile_end_add_data(&dasd_global_profile_data, 832 if (dasd_global_profile.data) {
833 dasd_profile_end_add_data(dasd_global_profile.data,
830 cqr->startdev != block->base, 834 cqr->startdev != block->base,
831 cqr->cpmode == 1, 835 cqr->cpmode == 1,
832 rq_data_dir(req) == READ, 836 rq_data_dir(req) == READ,
@@ -835,6 +839,7 @@ static void dasd_profile_end(struct dasd_block *block,
835 irqtime_ind, irqtimeps_ind, 839 irqtime_ind, irqtimeps_ind,
836 endtime_ind); 840 endtime_ind);
837 } 841 }
842 spin_unlock(&dasd_global_profile.lock);
838 843
839 spin_lock(&block->profile.lock); 844 spin_lock(&block->profile.lock);
840 if (block->profile.data) 845 if (block->profile.data)
@@ -876,12 +881,6 @@ void dasd_profile_reset(struct dasd_profile *profile)
876 spin_unlock_bh(&profile->lock); 881 spin_unlock_bh(&profile->lock);
877} 882}
878 883
879void dasd_global_profile_reset(void)
880{
881 memset(&dasd_global_profile_data, 0, sizeof(dasd_global_profile_data));
882 getnstimeofday(&dasd_global_profile_data.starttod);
883}
884
885int dasd_profile_on(struct dasd_profile *profile) 884int dasd_profile_on(struct dasd_profile *profile)
886{ 885{
887 struct dasd_profile_info *data; 886 struct dasd_profile_info *data;
@@ -949,12 +948,20 @@ static ssize_t dasd_stats_write(struct file *file,
949 dasd_profile_reset(prof); 948 dasd_profile_reset(prof);
950 } else if (strncmp(str, "on", 2) == 0) { 949 } else if (strncmp(str, "on", 2) == 0) {
951 rc = dasd_profile_on(prof); 950 rc = dasd_profile_on(prof);
952 if (!rc) 951 if (rc)
953 rc = user_len; 952 goto out;
953 rc = user_len;
954 if (prof == &dasd_global_profile) {
955 dasd_profile_reset(prof);
956 dasd_global_profile_level = DASD_PROFILE_GLOBAL_ONLY;
957 }
954 } else if (strncmp(str, "off", 3) == 0) { 958 } else if (strncmp(str, "off", 3) == 0) {
959 if (prof == &dasd_global_profile)
960 dasd_global_profile_level = DASD_PROFILE_OFF;
955 dasd_profile_off(prof); 961 dasd_profile_off(prof);
956 } else 962 } else
957 rc = -EINVAL; 963 rc = -EINVAL;
964out:
958 vfree(buffer); 965 vfree(buffer);
959 return rc; 966 return rc;
960} 967}
@@ -1044,57 +1051,6 @@ static const struct file_operations dasd_stats_raw_fops = {
1044 .write = dasd_stats_write, 1051 .write = dasd_stats_write,
1045}; 1052};
1046 1053
1047static ssize_t dasd_stats_global_write(struct file *file,
1048 const char __user *user_buf,
1049 size_t user_len, loff_t *pos)
1050{
1051 char *buffer, *str;
1052 ssize_t rc;
1053
1054 if (user_len > 65536)
1055 user_len = 65536;
1056 buffer = dasd_get_user_string(user_buf, user_len);
1057 if (IS_ERR(buffer))
1058 return PTR_ERR(buffer);
1059 str = skip_spaces(buffer);
1060 rc = user_len;
1061 if (strncmp(str, "reset", 5) == 0) {
1062 dasd_global_profile_reset();
1063 } else if (strncmp(str, "on", 2) == 0) {
1064 dasd_global_profile_reset();
1065 dasd_global_profile_level = DASD_PROFILE_GLOBAL_ONLY;
1066 } else if (strncmp(str, "off", 3) == 0) {
1067 dasd_global_profile_level = DASD_PROFILE_OFF;
1068 } else
1069 rc = -EINVAL;
1070 vfree(buffer);
1071 return rc;
1072}
1073
1074static int dasd_stats_global_show(struct seq_file *m, void *v)
1075{
1076 if (!dasd_global_profile_level) {
1077 seq_puts(m, "disabled\n");
1078 return 0;
1079 }
1080 dasd_stats_seq_print(m, &dasd_global_profile_data);
1081 return 0;
1082}
1083
1084static int dasd_stats_global_open(struct inode *inode, struct file *file)
1085{
1086 return single_open(file, dasd_stats_global_show, NULL);
1087}
1088
1089static const struct file_operations dasd_stats_global_fops = {
1090 .owner = THIS_MODULE,
1091 .open = dasd_stats_global_open,
1092 .read = seq_read,
1093 .llseek = seq_lseek,
1094 .release = single_release,
1095 .write = dasd_stats_global_write,
1096};
1097
1098static void dasd_profile_init(struct dasd_profile *profile, 1054static void dasd_profile_init(struct dasd_profile *profile,
1099 struct dentry *base_dentry) 1055 struct dentry *base_dentry)
1100{ 1056{
@@ -1123,20 +1079,16 @@ static void dasd_profile_exit(struct dasd_profile *profile)
1123static void dasd_statistics_removeroot(void) 1079static void dasd_statistics_removeroot(void)
1124{ 1080{
1125 dasd_global_profile_level = DASD_PROFILE_OFF; 1081 dasd_global_profile_level = DASD_PROFILE_OFF;
1126 debugfs_remove(dasd_global_profile_dentry); 1082 dasd_profile_exit(&dasd_global_profile);
1127 dasd_global_profile_dentry = NULL;
1128 debugfs_remove(dasd_debugfs_global_entry); 1083 debugfs_remove(dasd_debugfs_global_entry);
1129 debugfs_remove(dasd_debugfs_root_entry); 1084 debugfs_remove(dasd_debugfs_root_entry);
1130} 1085}
1131 1086
1132static void dasd_statistics_createroot(void) 1087static void dasd_statistics_createroot(void)
1133{ 1088{
1134 umode_t mode;
1135 struct dentry *pde; 1089 struct dentry *pde;
1136 1090
1137 dasd_debugfs_root_entry = NULL; 1091 dasd_debugfs_root_entry = NULL;
1138 dasd_debugfs_global_entry = NULL;
1139 dasd_global_profile_dentry = NULL;
1140 pde = debugfs_create_dir("dasd", NULL); 1092 pde = debugfs_create_dir("dasd", NULL);
1141 if (!pde || IS_ERR(pde)) 1093 if (!pde || IS_ERR(pde))
1142 goto error; 1094 goto error;
@@ -1145,13 +1097,7 @@ static void dasd_statistics_createroot(void)
1145 if (!pde || IS_ERR(pde)) 1097 if (!pde || IS_ERR(pde))
1146 goto error; 1098 goto error;
1147 dasd_debugfs_global_entry = pde; 1099 dasd_debugfs_global_entry = pde;
1148 1100 dasd_profile_init(&dasd_global_profile, dasd_debugfs_global_entry);
1149 mode = (S_IRUSR | S_IWUSR | S_IFREG);
1150 pde = debugfs_create_file("statistics", mode, dasd_debugfs_global_entry,
1151 NULL, &dasd_stats_global_fops);
1152 if (!pde || IS_ERR(pde))
1153 goto error;
1154 dasd_global_profile_dentry = pde;
1155 return; 1101 return;
1156 1102
1157error: 1103error:
diff --git a/drivers/s390/block/dasd_int.h b/drivers/s390/block/dasd_int.h
index 8b5d4100abf7..227e3dea3155 100644
--- a/drivers/s390/block/dasd_int.h
+++ b/drivers/s390/block/dasd_int.h
@@ -651,7 +651,7 @@ dasd_check_blocksize(int bsize)
651#define DASD_PROFILE_GLOBAL_ONLY 2 651#define DASD_PROFILE_GLOBAL_ONLY 2
652 652
653extern debug_info_t *dasd_debug_area; 653extern debug_info_t *dasd_debug_area;
654extern struct dasd_profile_info dasd_global_profile_data; 654extern struct dasd_profile dasd_global_profile;
655extern unsigned int dasd_global_profile_level; 655extern unsigned int dasd_global_profile_level;
656extern const struct block_device_operations dasd_device_operations; 656extern const struct block_device_operations dasd_device_operations;
657 657
@@ -728,7 +728,6 @@ int dasd_device_is_ro(struct dasd_device *);
728void dasd_profile_reset(struct dasd_profile *); 728void dasd_profile_reset(struct dasd_profile *);
729int dasd_profile_on(struct dasd_profile *); 729int dasd_profile_on(struct dasd_profile *);
730void dasd_profile_off(struct dasd_profile *); 730void dasd_profile_off(struct dasd_profile *);
731void dasd_global_profile_reset(void);
732char *dasd_get_user_string(const char __user *, size_t); 731char *dasd_get_user_string(const char __user *, size_t);
733 732
734/* externals in dasd_devmap.c */ 733/* externals in dasd_devmap.c */
diff --git a/drivers/s390/block/dasd_proc.c b/drivers/s390/block/dasd_proc.c
index 78ac905a5b7f..aa7bb2d1da81 100644
--- a/drivers/s390/block/dasd_proc.c
+++ b/drivers/s390/block/dasd_proc.c
@@ -212,14 +212,15 @@ static int dasd_stats_proc_show(struct seq_file *m, void *v)
212 struct dasd_profile_info *prof; 212 struct dasd_profile_info *prof;
213 int factor; 213 int factor;
214 214
215 /* check for active profiling */ 215 spin_lock_bh(&dasd_global_profile.lock);
216 if (!dasd_global_profile_level) { 216 prof = dasd_global_profile.data;
217 if (!prof) {
218 spin_unlock_bh(&dasd_global_profile.lock);
217 seq_printf(m, "Statistics are off - they might be " 219 seq_printf(m, "Statistics are off - they might be "
218 "switched on using 'echo set on > " 220 "switched on using 'echo set on > "
219 "/proc/dasd/statistics'\n"); 221 "/proc/dasd/statistics'\n");
220 return 0; 222 return 0;
221 } 223 }
222 prof = &dasd_global_profile_data;
223 224
224 /* prevent counter 'overflow' on output */ 225 /* prevent counter 'overflow' on output */
225 for (factor = 1; (prof->dasd_io_reqs / factor) > 9999999; 226 for (factor = 1; (prof->dasd_io_reqs / factor) > 9999999;
@@ -255,6 +256,7 @@ static int dasd_stats_proc_show(struct seq_file *m, void *v)
255 dasd_statistics_array(m, prof->dasd_io_time3, factor); 256 dasd_statistics_array(m, prof->dasd_io_time3, factor);
256 seq_printf(m, "# of req in chanq at enqueuing (1..32) \n"); 257 seq_printf(m, "# of req in chanq at enqueuing (1..32) \n");
257 dasd_statistics_array(m, prof->dasd_io_nr_req, factor); 258 dasd_statistics_array(m, prof->dasd_io_nr_req, factor);
259 spin_unlock_bh(&dasd_global_profile.lock);
258#else 260#else
259 seq_printf(m, "Statistics are not activated in this kernel\n"); 261 seq_printf(m, "Statistics are not activated in this kernel\n");
260#endif 262#endif
@@ -291,14 +293,19 @@ static ssize_t dasd_stats_proc_write(struct file *file,
291 dasd_stats_all_block_off(); 293 dasd_stats_all_block_off();
292 goto out_error; 294 goto out_error;
293 } 295 }
294 dasd_global_profile_reset(); 296 rc = dasd_profile_on(&dasd_global_profile);
297 if (rc) {
298 dasd_stats_all_block_off();
299 goto out_error;
300 }
301 dasd_profile_reset(&dasd_global_profile);
295 dasd_global_profile_level = DASD_PROFILE_ON; 302 dasd_global_profile_level = DASD_PROFILE_ON;
296 pr_info("The statistics feature has been switched " 303 pr_info("The statistics feature has been switched "
297 "on\n"); 304 "on\n");
298 } else if (strcmp(str, "off") == 0) { 305 } else if (strcmp(str, "off") == 0) {
299 /* switch off and reset statistics profiling */ 306 /* switch off statistics profiling */
300 dasd_global_profile_level = DASD_PROFILE_OFF; 307 dasd_global_profile_level = DASD_PROFILE_OFF;
301 dasd_global_profile_reset(); 308 dasd_profile_off(&dasd_global_profile);
302 dasd_stats_all_block_off(); 309 dasd_stats_all_block_off();
303 pr_info("The statistics feature has been switched " 310 pr_info("The statistics feature has been switched "
304 "off\n"); 311 "off\n");
@@ -306,7 +313,7 @@ static ssize_t dasd_stats_proc_write(struct file *file,
306 goto out_parse_error; 313 goto out_parse_error;
307 } else if (strncmp(str, "reset", 5) == 0) { 314 } else if (strncmp(str, "reset", 5) == 0) {
308 /* reset the statistics */ 315 /* reset the statistics */
309 dasd_global_profile_reset(); 316 dasd_profile_reset(&dasd_global_profile);
310 dasd_stats_all_block_reset(); 317 dasd_stats_all_block_reset();
311 pr_info("The statistics have been reset\n"); 318 pr_info("The statistics have been reset\n");
312 } else 319 } else
diff --git a/drivers/s390/block/dcssblk.c b/drivers/s390/block/dcssblk.c
index b550c8c8d010..7f900229404d 100644
--- a/drivers/s390/block/dcssblk.c
+++ b/drivers/s390/block/dcssblk.c
@@ -438,7 +438,13 @@ dcssblk_save_store(struct device *dev, struct device_attribute *attr, const char
438 pr_info("All DCSSs that map to device %s are " 438 pr_info("All DCSSs that map to device %s are "
439 "saved\n", dev_info->segment_name); 439 "saved\n", dev_info->segment_name);
440 list_for_each_entry(entry, &dev_info->seg_list, lh) { 440 list_for_each_entry(entry, &dev_info->seg_list, lh) {
441 segment_save(entry->segment_name); 441 if (entry->segment_type == SEG_TYPE_EN ||
442 entry->segment_type == SEG_TYPE_SN)
443 pr_warn("DCSS %s is of type SN or EN"
444 " and cannot be saved\n",
445 entry->segment_name);
446 else
447 segment_save(entry->segment_name);
442 } 448 }
443 } else { 449 } else {
444 // device is busy => we save it when it becomes 450 // device is busy => we save it when it becomes
@@ -797,7 +803,12 @@ dcssblk_release(struct gendisk *disk, fmode_t mode)
797 pr_info("Device %s has become idle and is being saved " 803 pr_info("Device %s has become idle and is being saved "
798 "now\n", dev_info->segment_name); 804 "now\n", dev_info->segment_name);
799 list_for_each_entry(entry, &dev_info->seg_list, lh) { 805 list_for_each_entry(entry, &dev_info->seg_list, lh) {
800 segment_save(entry->segment_name); 806 if (entry->segment_type == SEG_TYPE_EN ||
807 entry->segment_type == SEG_TYPE_SN)
808 pr_warn("DCSS %s is of type SN or EN and cannot"
809 " be saved\n", entry->segment_name);
810 else
811 segment_save(entry->segment_name);
801 } 812 }
802 dev_info->save_pending = 0; 813 dev_info->save_pending = 0;
803 } 814 }
diff --git a/drivers/s390/char/hmcdrv_ftp.c b/drivers/s390/char/hmcdrv_ftp.c
index 4bd63322fc29..d4b61d9088fb 100644
--- a/drivers/s390/char/hmcdrv_ftp.c
+++ b/drivers/s390/char/hmcdrv_ftp.c
@@ -200,10 +200,9 @@ int hmcdrv_ftp_probe(void)
200 rc = hmcdrv_ftp_startup(); 200 rc = hmcdrv_ftp_startup();
201 201
202 if (rc) 202 if (rc)
203 return rc; 203 goto out;
204 204
205 rc = hmcdrv_ftp_do(&ftp); 205 rc = hmcdrv_ftp_do(&ftp);
206 free_page((unsigned long) ftp.buf);
207 hmcdrv_ftp_shutdown(); 206 hmcdrv_ftp_shutdown();
208 207
209 switch (rc) { 208 switch (rc) {
@@ -216,7 +215,8 @@ int hmcdrv_ftp_probe(void)
216 rc = 0; /* clear length (success) */ 215 rc = 0; /* clear length (success) */
217 break; 216 break;
218 } /* switch */ 217 } /* switch */
219 218out:
219 free_page((unsigned long) ftp.buf);
220 return rc; 220 return rc;
221} 221}
222EXPORT_SYMBOL(hmcdrv_ftp_probe); 222EXPORT_SYMBOL(hmcdrv_ftp_probe);
diff --git a/drivers/s390/char/hmcdrv_mod.c b/drivers/s390/char/hmcdrv_mod.c
index 505c6a78ee1a..251a318a9b75 100644
--- a/drivers/s390/char/hmcdrv_mod.c
+++ b/drivers/s390/char/hmcdrv_mod.c
@@ -11,7 +11,6 @@
11#include <linux/kernel.h> 11#include <linux/kernel.h>
12#include <linux/module.h> 12#include <linux/module.h>
13#include <linux/moduleparam.h> 13#include <linux/moduleparam.h>
14#include <linux/version.h>
15#include <linux/stat.h> 14#include <linux/stat.h>
16 15
17#include "hmcdrv_ftp.h" 16#include "hmcdrv_ftp.h"
diff --git a/drivers/s390/char/sclp_early.c b/drivers/s390/char/sclp_early.c
index 5bd6cb145a87..daf6cd5079ec 100644
--- a/drivers/s390/char/sclp_early.c
+++ b/drivers/s390/char/sclp_early.c
@@ -20,26 +20,31 @@ struct read_info_sccb {
20 struct sccb_header header; /* 0-7 */ 20 struct sccb_header header; /* 0-7 */
21 u16 rnmax; /* 8-9 */ 21 u16 rnmax; /* 8-9 */
22 u8 rnsize; /* 10 */ 22 u8 rnsize; /* 10 */
23 u8 _reserved0[16 - 11]; /* 11-15 */ 23 u8 _pad_11[16 - 11]; /* 11-15 */
24 u16 ncpurl; /* 16-17 */ 24 u16 ncpurl; /* 16-17 */
25 u16 cpuoff; /* 18-19 */ 25 u16 cpuoff; /* 18-19 */
26 u8 _reserved7[24 - 20]; /* 20-23 */ 26 u8 _pad_20[24 - 20]; /* 20-23 */
27 u8 loadparm[8]; /* 24-31 */ 27 u8 loadparm[8]; /* 24-31 */
28 u8 _reserved1[48 - 32]; /* 32-47 */ 28 u8 _pad_32[42 - 32]; /* 32-41 */
29 u8 fac42; /* 42 */
30 u8 fac43; /* 43 */
31 u8 _pad_44[48 - 44]; /* 44-47 */
29 u64 facilities; /* 48-55 */ 32 u64 facilities; /* 48-55 */
30 u8 _reserved2a[76 - 56]; /* 56-75 */ 33 u8 _pad_56[66 - 56]; /* 56-65 */
34 u8 fac66; /* 66 */
35 u8 _pad_67[76 - 67]; /* 67-83 */
31 u32 ibc; /* 76-79 */ 36 u32 ibc; /* 76-79 */
32 u8 _reserved2b[84 - 80]; /* 80-83 */ 37 u8 _pad80[84 - 80]; /* 80-83 */
33 u8 fac84; /* 84 */ 38 u8 fac84; /* 84 */
34 u8 fac85; /* 85 */ 39 u8 fac85; /* 85 */
35 u8 _reserved3[91 - 86]; /* 86-90 */ 40 u8 _pad_86[91 - 86]; /* 86-90 */
36 u8 flags; /* 91 */ 41 u8 flags; /* 91 */
37 u8 _reserved4[100 - 92]; /* 92-99 */ 42 u8 _pad_92[100 - 92]; /* 92-99 */
38 u32 rnsize2; /* 100-103 */ 43 u32 rnsize2; /* 100-103 */
39 u64 rnmax2; /* 104-111 */ 44 u64 rnmax2; /* 104-111 */
40 u8 _reserved5[120 - 112]; /* 112-119 */ 45 u8 _pad_112[120 - 112]; /* 112-119 */
41 u16 hcpua; /* 120-121 */ 46 u16 hcpua; /* 120-121 */
42 u8 _reserved6[4096 - 122]; /* 122-4095 */ 47 u8 _pad_122[4096 - 122]; /* 122-4095 */
43} __packed __aligned(PAGE_SIZE); 48} __packed __aligned(PAGE_SIZE);
44 49
45static char sccb_early[PAGE_SIZE] __aligned(PAGE_SIZE) __initdata; 50static char sccb_early[PAGE_SIZE] __aligned(PAGE_SIZE) __initdata;
@@ -50,6 +55,10 @@ static unsigned int sclp_max_cpu;
50static struct sclp_ipl_info sclp_ipl_info; 55static struct sclp_ipl_info sclp_ipl_info;
51static unsigned char sclp_siif; 56static unsigned char sclp_siif;
52static u32 sclp_ibc; 57static u32 sclp_ibc;
58static unsigned int sclp_mtid;
59static unsigned int sclp_mtid_cp;
60static unsigned int sclp_mtid_max;
61static unsigned int sclp_mtid_prev;
53 62
54u64 sclp_facilities; 63u64 sclp_facilities;
55u8 sclp_fac84; 64u8 sclp_fac84;
@@ -128,7 +137,7 @@ static void __init sclp_facilities_detect(struct read_info_sccb *sccb)
128 boot_cpu_address = stap(); 137 boot_cpu_address = stap();
129 cpue = (void *)sccb + sccb->cpuoff; 138 cpue = (void *)sccb + sccb->cpuoff;
130 for (cpu = 0; cpu < sccb->ncpurl; cpue++, cpu++) { 139 for (cpu = 0; cpu < sccb->ncpurl; cpue++, cpu++) {
131 if (boot_cpu_address != cpue->address) 140 if (boot_cpu_address != cpue->core_id)
132 continue; 141 continue;
133 sclp_siif = cpue->siif; 142 sclp_siif = cpue->siif;
134 break; 143 break;
@@ -139,6 +148,11 @@ static void __init sclp_facilities_detect(struct read_info_sccb *sccb)
139 if (sccb->flags & 0x2) 148 if (sccb->flags & 0x2)
140 sclp_ipl_info.has_dump = 1; 149 sclp_ipl_info.has_dump = 1;
141 memcpy(&sclp_ipl_info.loadparm, &sccb->loadparm, LOADPARM_LEN); 150 memcpy(&sclp_ipl_info.loadparm, &sccb->loadparm, LOADPARM_LEN);
151
152 sclp_mtid = (sccb->fac42 & 0x80) ? (sccb->fac42 & 31) : 0;
153 sclp_mtid_cp = (sccb->fac42 & 0x80) ? (sccb->fac43 & 31) : 0;
154 sclp_mtid_max = max(sclp_mtid, sclp_mtid_cp);
155 sclp_mtid_prev = (sccb->fac42 & 0x80) ? (sccb->fac66 & 31) : 0;
142} 156}
143 157
144bool __init sclp_has_linemode(void) 158bool __init sclp_has_linemode(void)
@@ -178,6 +192,21 @@ unsigned int sclp_get_ibc(void)
178} 192}
179EXPORT_SYMBOL(sclp_get_ibc); 193EXPORT_SYMBOL(sclp_get_ibc);
180 194
195unsigned int sclp_get_mtid(u8 cpu_type)
196{
197 return cpu_type ? sclp_mtid : sclp_mtid_cp;
198}
199
200unsigned int sclp_get_mtid_max(void)
201{
202 return sclp_mtid_max;
203}
204
205unsigned int sclp_get_mtid_prev(void)
206{
207 return sclp_mtid_prev;
208}
209
181/* 210/*
182 * This function will be called after sclp_facilities_detect(), which gets 211 * This function will be called after sclp_facilities_detect(), which gets
183 * called from early.c code. The sclp_facilities_detect() function retrieves 212 * called from early.c code. The sclp_facilities_detect() function retrieves
diff --git a/drivers/s390/char/tape_34xx.c b/drivers/s390/char/tape_34xx.c
index 9aa79702b370..de69f0ddc321 100644
--- a/drivers/s390/char/tape_34xx.c
+++ b/drivers/s390/char/tape_34xx.c
@@ -773,13 +773,11 @@ tape_34xx_unit_check(struct tape_device *device, struct tape_request *request,
773 "occurred\n"); 773 "occurred\n");
774 return tape_34xx_erp_failed(request, -EIO); 774 return tape_34xx_erp_failed(request, -EIO);
775 case 0x57: 775 case 0x57:
776 if (device->cdev->id.driver_info == tape_3480) { 776 /*
777 /* Attention intercept. */ 777 * 3480: Attention intercept.
778 return tape_34xx_erp_retry(request); 778 * 3490: Global status intercept.
779 } else { 779 */
780 /* Global status intercept. */ 780 return tape_34xx_erp_retry(request);
781 return tape_34xx_erp_retry(request);
782 }
783 case 0x5a: 781 case 0x5a:
784 /* 782 /*
785 * Tape length incompatible. The tape inserted is too long, 783 * Tape length incompatible. The tape inserted is too long,
diff --git a/drivers/s390/cio/cio.c b/drivers/s390/cio/cio.c
index 10eb738fc81a..3578105989a0 100644
--- a/drivers/s390/cio/cio.c
+++ b/drivers/s390/cio/cio.c
@@ -938,7 +938,7 @@ void reipl_ccw_dev(struct ccw_dev_id *devid)
938{ 938{
939 struct subchannel_id uninitialized_var(schid); 939 struct subchannel_id uninitialized_var(schid);
940 940
941 s390_reset_system(NULL, NULL); 941 s390_reset_system(NULL, NULL, NULL);
942 if (reipl_find_schid(devid, &schid) != 0) 942 if (reipl_find_schid(devid, &schid) != 0)
943 panic("IPL Device not found\n"); 943 panic("IPL Device not found\n");
944 do_reipl_asm(*((__u32*)&schid)); 944 do_reipl_asm(*((__u32*)&schid));
diff --git a/drivers/s390/cio/idset.c b/drivers/s390/cio/idset.c
index 5a999084a229..b3e06a7b9480 100644
--- a/drivers/s390/cio/idset.c
+++ b/drivers/s390/cio/idset.c
@@ -38,11 +38,6 @@ void idset_free(struct idset *set)
38 vfree(set); 38 vfree(set);
39} 39}
40 40
41void idset_clear(struct idset *set)
42{
43 memset(set->bitmap, 0, bitmap_size(set->num_ssid, set->num_id));
44}
45
46void idset_fill(struct idset *set) 41void idset_fill(struct idset *set)
47{ 42{
48 memset(set->bitmap, 0xff, bitmap_size(set->num_ssid, set->num_id)); 43 memset(set->bitmap, 0xff, bitmap_size(set->num_ssid, set->num_id));
@@ -103,21 +98,6 @@ int idset_sch_contains(struct idset *set, struct subchannel_id schid)
103 return idset_contains(set, schid.ssid, schid.sch_no); 98 return idset_contains(set, schid.ssid, schid.sch_no);
104} 99}
105 100
106int idset_sch_get_first(struct idset *set, struct subchannel_id *schid)
107{
108 int ssid = 0;
109 int id = 0;
110 int rc;
111
112 rc = idset_get_first(set, &ssid, &id);
113 if (rc) {
114 init_subchannel_id(schid);
115 schid->ssid = ssid;
116 schid->sch_no = id;
117 }
118 return rc;
119}
120
121int idset_is_empty(struct idset *set) 101int idset_is_empty(struct idset *set)
122{ 102{
123 return bitmap_empty(set->bitmap, set->num_ssid * set->num_id); 103 return bitmap_empty(set->bitmap, set->num_ssid * set->num_id);
diff --git a/drivers/s390/cio/idset.h b/drivers/s390/cio/idset.h
index 06d3bc01bb09..22b58104683b 100644
--- a/drivers/s390/cio/idset.h
+++ b/drivers/s390/cio/idset.h
@@ -11,7 +11,6 @@
11struct idset; 11struct idset;
12 12
13void idset_free(struct idset *set); 13void idset_free(struct idset *set);
14void idset_clear(struct idset *set);
15void idset_fill(struct idset *set); 14void idset_fill(struct idset *set);
16 15
17struct idset *idset_sch_new(void); 16struct idset *idset_sch_new(void);
@@ -19,7 +18,6 @@ void idset_sch_add(struct idset *set, struct subchannel_id id);
19void idset_sch_del(struct idset *set, struct subchannel_id id); 18void idset_sch_del(struct idset *set, struct subchannel_id id);
20void idset_sch_del_subseq(struct idset *set, struct subchannel_id schid); 19void idset_sch_del_subseq(struct idset *set, struct subchannel_id schid);
21int idset_sch_contains(struct idset *set, struct subchannel_id id); 20int idset_sch_contains(struct idset *set, struct subchannel_id id);
22int idset_sch_get_first(struct idset *set, struct subchannel_id *id);
23int idset_is_empty(struct idset *set); 21int idset_is_empty(struct idset *set);
24void idset_add_set(struct idset *to, struct idset *from); 22void idset_add_set(struct idset *to, struct idset *from);
25 23
diff --git a/drivers/s390/crypto/ap_bus.c b/drivers/s390/crypto/ap_bus.c
index 4d41bf75c233..3d7f19fb9a4e 100644
--- a/drivers/s390/crypto/ap_bus.c
+++ b/drivers/s390/crypto/ap_bus.c
@@ -204,6 +204,24 @@ ap_test_queue(ap_qid_t qid, int *queue_depth, int *device_type)
204} 204}
205 205
206/** 206/**
207 * ap_query_facilities(): PQAP(TAPQ) query facilities.
208 * @qid: The AP queue number
209 *
210 * Returns content of general register 2 after the PQAP(TAPQ)
211 * instruction was called.
212 */
213static inline unsigned long ap_query_facilities(ap_qid_t qid)
214{
215 register unsigned long reg0 asm ("0") = qid | 0x00800000UL;
216 register unsigned long reg1 asm ("1");
217 register unsigned long reg2 asm ("2") = 0UL;
218
219 asm volatile(".long 0xb2af0000" /* PQAP(TAPQ) */
220 : "+d" (reg0), "=d" (reg1), "+d" (reg2) : : "cc");
221 return reg2;
222}
223
224/**
207 * ap_reset_queue(): Reset adjunct processor queue. 225 * ap_reset_queue(): Reset adjunct processor queue.
208 * @qid: The AP queue number 226 * @qid: The AP queue number
209 * 227 *
@@ -1007,6 +1025,51 @@ void ap_bus_force_rescan(void)
1007EXPORT_SYMBOL(ap_bus_force_rescan); 1025EXPORT_SYMBOL(ap_bus_force_rescan);
1008 1026
1009/* 1027/*
1028 * ap_test_config(): helper function to extract the nrth bit
1029 * within the unsigned int array field.
1030 */
1031static inline int ap_test_config(unsigned int *field, unsigned int nr)
1032{
1033 if (nr > 0xFFu)
1034 return 0;
1035 return ap_test_bit((field + (nr >> 5)), (nr & 0x1f));
1036}
1037
1038/*
1039 * ap_test_config_card_id(): Test, whether an AP card ID is configured.
1040 * @id AP card ID
1041 *
1042 * Returns 0 if the card is not configured
1043 * 1 if the card is configured or
1044 * if the configuration information is not available
1045 */
1046static inline int ap_test_config_card_id(unsigned int id)
1047{
1048 if (!ap_configuration)
1049 return 1;
1050 return ap_test_config(ap_configuration->apm, id);
1051}
1052
1053/*
1054 * ap_test_config_domain(): Test, whether an AP usage domain is configured.
1055 * @domain AP usage domain ID
1056 *
1057 * Returns 0 if the usage domain is not configured
1058 * 1 if the usage domain is configured or
1059 * if the configuration information is not available
1060 */
1061static inline int ap_test_config_domain(unsigned int domain)
1062{
1063 if (!ap_configuration) /* QCI not supported */
1064 if (domain < 16)
1065 return 1; /* then domains 0...15 are configured */
1066 else
1067 return 0;
1068 else
1069 return ap_test_config(ap_configuration->aqm, domain);
1070}
1071
1072/*
1010 * AP bus attributes. 1073 * AP bus attributes.
1011 */ 1074 */
1012static ssize_t ap_domain_show(struct bus_type *bus, char *buf) 1075static ssize_t ap_domain_show(struct bus_type *bus, char *buf)
@@ -1121,6 +1184,42 @@ static ssize_t poll_timeout_store(struct bus_type *bus, const char *buf,
1121 1184
1122static BUS_ATTR(poll_timeout, 0644, poll_timeout_show, poll_timeout_store); 1185static BUS_ATTR(poll_timeout, 0644, poll_timeout_show, poll_timeout_store);
1123 1186
1187static ssize_t ap_max_domain_id_show(struct bus_type *bus, char *buf)
1188{
1189 ap_qid_t qid;
1190 int i, nd, max_domain_id = -1;
1191 unsigned long fbits;
1192
1193 if (ap_configuration) {
1194 if (ap_domain_index >= 0 && ap_domain_index < AP_DOMAINS) {
1195 for (i = 0; i < AP_DEVICES; i++) {
1196 if (!ap_test_config_card_id(i))
1197 continue;
1198 qid = AP_MKQID(i, ap_domain_index);
1199 fbits = ap_query_facilities(qid);
1200 if (fbits & (1UL << 57)) {
1201 /* the N bit is 0, Nd field is filled */
1202 nd = (int)((fbits & 0x00FF0000UL)>>16);
1203 if (nd > 0)
1204 max_domain_id = nd;
1205 else
1206 max_domain_id = 15;
1207 } else {
1208 /* N bit is 1, max 16 domains */
1209 max_domain_id = 15;
1210 }
1211 break;
1212 }
1213 }
1214 } else {
1215 /* no APXA support, older machines with max 16 domains */
1216 max_domain_id = 15;
1217 }
1218 return snprintf(buf, PAGE_SIZE, "%d\n", max_domain_id);
1219}
1220
1221static BUS_ATTR(ap_max_domain_id, 0444, ap_max_domain_id_show, NULL);
1222
1124static struct bus_attribute *const ap_bus_attrs[] = { 1223static struct bus_attribute *const ap_bus_attrs[] = {
1125 &bus_attr_ap_domain, 1224 &bus_attr_ap_domain,
1126 &bus_attr_ap_control_domain_mask, 1225 &bus_attr_ap_control_domain_mask,
@@ -1128,50 +1227,10 @@ static struct bus_attribute *const ap_bus_attrs[] = {
1128 &bus_attr_poll_thread, 1227 &bus_attr_poll_thread,
1129 &bus_attr_ap_interrupts, 1228 &bus_attr_ap_interrupts,
1130 &bus_attr_poll_timeout, 1229 &bus_attr_poll_timeout,
1230 &bus_attr_ap_max_domain_id,
1131 NULL, 1231 NULL,
1132}; 1232};
1133 1233
1134static inline int ap_test_config(unsigned int *field, unsigned int nr)
1135{
1136 if (nr > 0xFFu)
1137 return 0;
1138 return ap_test_bit((field + (nr >> 5)), (nr & 0x1f));
1139}
1140
1141/*
1142 * ap_test_config_card_id(): Test, whether an AP card ID is configured.
1143 * @id AP card ID
1144 *
1145 * Returns 0 if the card is not configured
1146 * 1 if the card is configured or
1147 * if the configuration information is not available
1148 */
1149static inline int ap_test_config_card_id(unsigned int id)
1150{
1151 if (!ap_configuration)
1152 return 1;
1153 return ap_test_config(ap_configuration->apm, id);
1154}
1155
1156/*
1157 * ap_test_config_domain(): Test, whether an AP usage domain is configured.
1158 * @domain AP usage domain ID
1159 *
1160 * Returns 0 if the usage domain is not configured
1161 * 1 if the usage domain is configured or
1162 * if the configuration information is not available
1163 */
1164static inline int ap_test_config_domain(unsigned int domain)
1165{
1166 if (!ap_configuration) /* QCI not supported */
1167 if (domain < 16)
1168 return 1; /* then domains 0...15 are configured */
1169 else
1170 return 0;
1171 else
1172 return ap_test_config(ap_configuration->aqm, domain);
1173}
1174
1175/** 1234/**
1176 * ap_query_configuration(): Query AP configuration information. 1235 * ap_query_configuration(): Query AP configuration information.
1177 * 1236 *
@@ -1434,9 +1493,6 @@ static void ap_scan_bus(struct work_struct *unused)
1434 continue; 1493 continue;
1435 } 1494 }
1436 break; 1495 break;
1437 case 11:
1438 ap_dev->device_type = 10;
1439 break;
1440 default: 1496 default:
1441 ap_dev->device_type = device_type; 1497 ap_dev->device_type = device_type;
1442 } 1498 }
diff --git a/drivers/s390/crypto/ap_bus.h b/drivers/s390/crypto/ap_bus.h
index 055a0f956d17..2737d261a324 100644
--- a/drivers/s390/crypto/ap_bus.h
+++ b/drivers/s390/crypto/ap_bus.h
@@ -117,6 +117,7 @@ static inline int ap_test_bit(unsigned int *ptr, unsigned int nr)
117#define AP_DEVICE_TYPE_CEX3A 8 117#define AP_DEVICE_TYPE_CEX3A 8
118#define AP_DEVICE_TYPE_CEX3C 9 118#define AP_DEVICE_TYPE_CEX3C 9
119#define AP_DEVICE_TYPE_CEX4 10 119#define AP_DEVICE_TYPE_CEX4 10
120#define AP_DEVICE_TYPE_CEX5 11
120 121
121/* 122/*
122 * Known function facilities 123 * Known function facilities
diff --git a/drivers/s390/crypto/zcrypt_api.h b/drivers/s390/crypto/zcrypt_api.h
index b3d496bfaa7e..750876891931 100644
--- a/drivers/s390/crypto/zcrypt_api.h
+++ b/drivers/s390/crypto/zcrypt_api.h
@@ -75,6 +75,7 @@ struct ica_z90_status {
75#define ZCRYPT_CEX3C 7 75#define ZCRYPT_CEX3C 7
76#define ZCRYPT_CEX3A 8 76#define ZCRYPT_CEX3A 8
77#define ZCRYPT_CEX4 10 77#define ZCRYPT_CEX4 10
78#define ZCRYPT_CEX5 11
78 79
79/** 80/**
80 * Large random numbers are pulled in 4096 byte chunks from the crypto cards 81 * Large random numbers are pulled in 4096 byte chunks from the crypto cards
diff --git a/drivers/s390/crypto/zcrypt_cex4.c b/drivers/s390/crypto/zcrypt_cex4.c
index 569f8b1d86c0..71e698b85772 100644
--- a/drivers/s390/crypto/zcrypt_cex4.c
+++ b/drivers/s390/crypto/zcrypt_cex4.c
@@ -26,6 +26,10 @@
26 26
27#define CEX4A_SPEED_RATING 900 /* TODO new card, new speed rating */ 27#define CEX4A_SPEED_RATING 900 /* TODO new card, new speed rating */
28#define CEX4C_SPEED_RATING 6500 /* TODO new card, new speed rating */ 28#define CEX4C_SPEED_RATING 6500 /* TODO new card, new speed rating */
29#define CEX4P_SPEED_RATING 7000 /* TODO new card, new speed rating */
30#define CEX5A_SPEED_RATING 450 /* TODO new card, new speed rating */
31#define CEX5C_SPEED_RATING 3250 /* TODO new card, new speed rating */
32#define CEX5P_SPEED_RATING 3500 /* TODO new card, new speed rating */
29 33
30#define CEX4A_MAX_MESSAGE_SIZE MSGTYPE50_CRB3_MAX_MSG_SIZE 34#define CEX4A_MAX_MESSAGE_SIZE MSGTYPE50_CRB3_MAX_MSG_SIZE
31#define CEX4C_MAX_MESSAGE_SIZE MSGTYPE06_MAX_MSG_SIZE 35#define CEX4C_MAX_MESSAGE_SIZE MSGTYPE06_MAX_MSG_SIZE
@@ -39,6 +43,7 @@
39 43
40static struct ap_device_id zcrypt_cex4_ids[] = { 44static struct ap_device_id zcrypt_cex4_ids[] = {
41 { AP_DEVICE(AP_DEVICE_TYPE_CEX4) }, 45 { AP_DEVICE(AP_DEVICE_TYPE_CEX4) },
46 { AP_DEVICE(AP_DEVICE_TYPE_CEX5) },
42 { /* end of list */ }, 47 { /* end of list */ },
43}; 48};
44 49
@@ -70,11 +75,18 @@ static int zcrypt_cex4_probe(struct ap_device *ap_dev)
70 75
71 switch (ap_dev->device_type) { 76 switch (ap_dev->device_type) {
72 case AP_DEVICE_TYPE_CEX4: 77 case AP_DEVICE_TYPE_CEX4:
78 case AP_DEVICE_TYPE_CEX5:
73 if (ap_test_bit(&ap_dev->functions, AP_FUNC_ACCEL)) { 79 if (ap_test_bit(&ap_dev->functions, AP_FUNC_ACCEL)) {
74 zdev = zcrypt_device_alloc(CEX4A_MAX_MESSAGE_SIZE); 80 zdev = zcrypt_device_alloc(CEX4A_MAX_MESSAGE_SIZE);
75 if (!zdev) 81 if (!zdev)
76 return -ENOMEM; 82 return -ENOMEM;
77 zdev->type_string = "CEX4A"; 83 if (ap_dev->device_type == AP_DEVICE_TYPE_CEX4) {
84 zdev->type_string = "CEX4A";
85 zdev->speed_rating = CEX4A_SPEED_RATING;
86 } else {
87 zdev->type_string = "CEX5A";
88 zdev->speed_rating = CEX5A_SPEED_RATING;
89 }
78 zdev->user_space_type = ZCRYPT_CEX3A; 90 zdev->user_space_type = ZCRYPT_CEX3A;
79 zdev->min_mod_size = CEX4A_MIN_MOD_SIZE; 91 zdev->min_mod_size = CEX4A_MIN_MOD_SIZE;
80 if (ap_test_bit(&ap_dev->functions, AP_FUNC_MEX4K) && 92 if (ap_test_bit(&ap_dev->functions, AP_FUNC_MEX4K) &&
@@ -90,33 +102,42 @@ static int zcrypt_cex4_probe(struct ap_device *ap_dev)
90 CEX4A_MAX_MOD_SIZE_2K; 102 CEX4A_MAX_MOD_SIZE_2K;
91 } 103 }
92 zdev->short_crt = 1; 104 zdev->short_crt = 1;
93 zdev->speed_rating = CEX4A_SPEED_RATING;
94 zdev->ops = zcrypt_msgtype_request(MSGTYPE50_NAME, 105 zdev->ops = zcrypt_msgtype_request(MSGTYPE50_NAME,
95 MSGTYPE50_VARIANT_DEFAULT); 106 MSGTYPE50_VARIANT_DEFAULT);
96 } else if (ap_test_bit(&ap_dev->functions, AP_FUNC_COPRO)) { 107 } else if (ap_test_bit(&ap_dev->functions, AP_FUNC_COPRO)) {
97 zdev = zcrypt_device_alloc(CEX4C_MAX_MESSAGE_SIZE); 108 zdev = zcrypt_device_alloc(CEX4C_MAX_MESSAGE_SIZE);
98 if (!zdev) 109 if (!zdev)
99 return -ENOMEM; 110 return -ENOMEM;
100 zdev->type_string = "CEX4C"; 111 if (ap_dev->device_type == AP_DEVICE_TYPE_CEX4) {
112 zdev->type_string = "CEX4C";
113 zdev->speed_rating = CEX4C_SPEED_RATING;
114 } else {
115 zdev->type_string = "CEX5C";
116 zdev->speed_rating = CEX5C_SPEED_RATING;
117 }
101 zdev->user_space_type = ZCRYPT_CEX3C; 118 zdev->user_space_type = ZCRYPT_CEX3C;
102 zdev->min_mod_size = CEX4C_MIN_MOD_SIZE; 119 zdev->min_mod_size = CEX4C_MIN_MOD_SIZE;
103 zdev->max_mod_size = CEX4C_MAX_MOD_SIZE; 120 zdev->max_mod_size = CEX4C_MAX_MOD_SIZE;
104 zdev->max_exp_bit_length = CEX4C_MAX_MOD_SIZE; 121 zdev->max_exp_bit_length = CEX4C_MAX_MOD_SIZE;
105 zdev->short_crt = 0; 122 zdev->short_crt = 0;
106 zdev->speed_rating = CEX4C_SPEED_RATING;
107 zdev->ops = zcrypt_msgtype_request(MSGTYPE06_NAME, 123 zdev->ops = zcrypt_msgtype_request(MSGTYPE06_NAME,
108 MSGTYPE06_VARIANT_DEFAULT); 124 MSGTYPE06_VARIANT_DEFAULT);
109 } else if (ap_test_bit(&ap_dev->functions, AP_FUNC_EP11)) { 125 } else if (ap_test_bit(&ap_dev->functions, AP_FUNC_EP11)) {
110 zdev = zcrypt_device_alloc(CEX4C_MAX_MESSAGE_SIZE); 126 zdev = zcrypt_device_alloc(CEX4C_MAX_MESSAGE_SIZE);
111 if (!zdev) 127 if (!zdev)
112 return -ENOMEM; 128 return -ENOMEM;
113 zdev->type_string = "CEX4P"; 129 if (ap_dev->device_type == AP_DEVICE_TYPE_CEX4) {
130 zdev->type_string = "CEX4P";
131 zdev->speed_rating = CEX4P_SPEED_RATING;
132 } else {
133 zdev->type_string = "CEX5P";
134 zdev->speed_rating = CEX5P_SPEED_RATING;
135 }
114 zdev->user_space_type = ZCRYPT_CEX4; 136 zdev->user_space_type = ZCRYPT_CEX4;
115 zdev->min_mod_size = CEX4C_MIN_MOD_SIZE; 137 zdev->min_mod_size = CEX4C_MIN_MOD_SIZE;
116 zdev->max_mod_size = CEX4C_MAX_MOD_SIZE; 138 zdev->max_mod_size = CEX4C_MAX_MOD_SIZE;
117 zdev->max_exp_bit_length = CEX4C_MAX_MOD_SIZE; 139 zdev->max_exp_bit_length = CEX4C_MAX_MOD_SIZE;
118 zdev->short_crt = 0; 140 zdev->short_crt = 0;
119 zdev->speed_rating = CEX4C_SPEED_RATING;
120 zdev->ops = zcrypt_msgtype_request(MSGTYPE06_NAME, 141 zdev->ops = zcrypt_msgtype_request(MSGTYPE06_NAME,
121 MSGTYPE06_VARIANT_EP11); 142 MSGTYPE06_VARIANT_EP11);
122 } 143 }
diff --git a/drivers/tty/hvc/hvc_iucv.c b/drivers/tty/hvc/hvc_iucv.c
index ea74460f3638..f78a87b07872 100644
--- a/drivers/tty/hvc/hvc_iucv.c
+++ b/drivers/tty/hvc/hvc_iucv.c
@@ -1,10 +1,10 @@
1/* 1/*
2 * hvc_iucv.c - z/VM IUCV hypervisor console (HVC) device driver 2 * z/VM IUCV hypervisor console (HVC) device driver
3 * 3 *
4 * This HVC device driver provides terminal access using 4 * This HVC device driver provides terminal access using
5 * z/VM IUCV communication paths. 5 * z/VM IUCV communication paths.
6 * 6 *
7 * Copyright IBM Corp. 2008, 2009 7 * Copyright IBM Corp. 2008, 2013
8 * 8 *
9 * Author(s): Hendrik Brueckner <brueckner@linux.vnet.ibm.com> 9 * Author(s): Hendrik Brueckner <brueckner@linux.vnet.ibm.com>
10 */ 10 */
@@ -102,6 +102,7 @@ static struct hvc_iucv_private *hvc_iucv_table[MAX_HVC_IUCV_LINES];
102#define IUCV_HVC_CON_IDX (0) 102#define IUCV_HVC_CON_IDX (0)
103/* List of z/VM user ID filter entries (struct iucv_vmid_filter) */ 103/* List of z/VM user ID filter entries (struct iucv_vmid_filter) */
104#define MAX_VMID_FILTER (500) 104#define MAX_VMID_FILTER (500)
105#define FILTER_WILDCARD_CHAR '*'
105static size_t hvc_iucv_filter_size; 106static size_t hvc_iucv_filter_size;
106static void *hvc_iucv_filter; 107static void *hvc_iucv_filter;
107static const char *hvc_iucv_filter_string; 108static const char *hvc_iucv_filter_string;
@@ -734,20 +735,31 @@ static void hvc_iucv_notifier_del(struct hvc_struct *hp, int id)
734 * hvc_iucv_filter_connreq() - Filter connection request based on z/VM user ID 735 * hvc_iucv_filter_connreq() - Filter connection request based on z/VM user ID
735 * @ipvmid: Originating z/VM user ID (right padded with blanks) 736 * @ipvmid: Originating z/VM user ID (right padded with blanks)
736 * 737 *
737 * Returns 0 if the z/VM user ID @ipvmid is allowed to connection, otherwise 738 * Returns 0 if the z/VM user ID that is specified with @ipvmid is permitted to
738 * non-zero. 739 * connect, otherwise non-zero.
739 */ 740 */
740static int hvc_iucv_filter_connreq(u8 ipvmid[8]) 741static int hvc_iucv_filter_connreq(u8 ipvmid[8])
741{ 742{
742 size_t i; 743 const char *wildcard, *filter_entry;
744 size_t i, len;
743 745
744 /* Note: default policy is ACCEPT if no filter is set */ 746 /* Note: default policy is ACCEPT if no filter is set */
745 if (!hvc_iucv_filter_size) 747 if (!hvc_iucv_filter_size)
746 return 0; 748 return 0;
747 749
748 for (i = 0; i < hvc_iucv_filter_size; i++) 750 for (i = 0; i < hvc_iucv_filter_size; i++) {
749 if (0 == memcmp(ipvmid, hvc_iucv_filter + (8 * i), 8)) 751 filter_entry = hvc_iucv_filter + (8 * i);
752
753 /* If a filter entry contains the filter wildcard character,
754 * reduce the length to match the leading portion of the user
755 * ID only (wildcard match). Characters following the wildcard
756 * are ignored.
757 */
758 wildcard = strnchr(filter_entry, 8, FILTER_WILDCARD_CHAR);
759 len = (wildcard) ? wildcard - filter_entry : 8;
760 if (0 == memcmp(ipvmid, filter_entry, len))
750 return 0; 761 return 0;
762 }
751 return 1; 763 return 1;
752} 764}
753 765
@@ -1166,6 +1178,7 @@ static void __init hvc_iucv_destroy(struct hvc_iucv_private *priv)
1166/** 1178/**
1167 * hvc_iucv_parse_filter() - Parse filter for a single z/VM user ID 1179 * hvc_iucv_parse_filter() - Parse filter for a single z/VM user ID
1168 * @filter: String containing a comma-separated list of z/VM user IDs 1180 * @filter: String containing a comma-separated list of z/VM user IDs
1181 * @dest: Location where to store the parsed z/VM user ID
1169 */ 1182 */
1170static const char *hvc_iucv_parse_filter(const char *filter, char *dest) 1183static const char *hvc_iucv_parse_filter(const char *filter, char *dest)
1171{ 1184{
@@ -1188,6 +1201,10 @@ static const char *hvc_iucv_parse_filter(const char *filter, char *dest)
1188 if (filter[len - 1] == '\n') 1201 if (filter[len - 1] == '\n')
1189 len--; 1202 len--;
1190 1203
1204 /* prohibit filter entries containing the wildcard character only */
1205 if (len == 1 && *filter == FILTER_WILDCARD_CHAR)
1206 return ERR_PTR(-EINVAL);
1207
1191 if (len > 8) 1208 if (len > 8)
1192 return ERR_PTR(-EINVAL); 1209 return ERR_PTR(-EINVAL);
1193 1210