Commit da8ac5e0 authored by Linus Torvalds's avatar Linus Torvalds

Merge branch 'for-linus' of git://git390.osdl.marist.edu/pub/scm/linux-2.6

* 'for-linus' of git://git390.osdl.marist.edu/pub/scm/linux-2.6: (38 commits)
  [S390] SPIN_LOCK_UNLOCKED cleanup in drivers/s390
  [S390] Clean up smp code in preparation for some larger changes.
  [S390] Remove debugging junk.
  [S390] Switch etr from tasklet to workqueue.
  [S390] split page_test_and_clear_dirty.
  [S390] Processor degradation notification.
  [S390] vtime: cleanup per_cpu usage.
  [S390] crypto: cleanup.
  [S390] sclp: fix coding style.
  [S390] vmlogrdr: stop IUCV connection in vmlogrdr_release.
  [S390] sclp: initialize early.
  [S390] ctc: kmalloc->kzalloc/casting cleanups.
  [S390] zfcpdump support.
  [S390] dasd: Add ipldev parameter.
  [S390] dasd: Add sysfs attribute status and generate uevents.
  [S390] Improved kernel stack overflow checking.
  [S390] Get rid of console setup functions.
  [S390] No execute support cleanup.
  [S390] Minor fault path optimization.
  [S390] Use generic bug.
  ...
parents 32f15dc5 cb629a01
crypto-API support for z990 Message Security Assist (MSA) instructions
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
AUTHOR: Thomas Spatzier (tspat@de.ibm.com)
1. Introduction crypto-API
~~~~~~~~~~~~~~~~~~~~~~~~~~
See Documentation/crypto/api-intro.txt for an introduction/description of the
kernel crypto API.
According to api-intro.txt support for z990 crypto instructions has been added
in the algorithm api layer of the crypto API. Several files containing z990
optimized implementations of crypto algorithms are placed in the
arch/s390/crypto directory.
2. Probing for availability of MSA
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
It should be possible to use Kernels with the z990 crypto implementations both
on machines with MSA available and on those without MSA (pre z990 or z990
without MSA). Therefore a simple probing mechanism has been implemented:
In the init function of each crypto module the availability of MSA and of the
respective crypto algorithm in particular will be tested. If the algorithm is
available the module will load and register its algorithm with the crypto API.
If the respective crypto algorithm is not available, the init function will
return -ENOSYS. In that case a fallback to the standard software implementation
of the crypto algorithm must be taken ( -> the standard crypto modules are
also built when compiling the kernel).
3. Ensuring z990 crypto module preference
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
If z990 crypto instructions are available the optimized modules should be
preferred instead of standard modules.
3.1. compiled-in modules
~~~~~~~~~~~~~~~~~~~~~~~~
For compiled-in modules it has to be ensured that the z990 modules are linked
before the standard crypto modules. Then, on system startup the init functions
of z990 crypto modules will be called first and query for availability of z990
crypto instructions. If instruction is available, the z990 module will register
its crypto algorithm implementation -> the load of the standard module will fail
since the algorithm is already registered.
If z990 crypto instruction is not available the load of the z990 module will
fail -> the standard module will load and register its algorithm.
3.2. dynamic modules
~~~~~~~~~~~~~~~~~~~~
A system administrator has to take care of giving preference to z990 crypto
modules. If MSA is available appropriate lines have to be added to
/etc/modprobe.conf.
Example: z990 crypto instruction for SHA1 algorithm is available
add the following line to /etc/modprobe.conf (assuming the
z990 crypto modules for SHA1 is called sha1_z990):
alias sha1 sha1_z990
-> when the sha1 algorithm is requested through the crypto API
(which has a module autoloader) the z990 module will be loaded.
TBD: a userspace module probing mechanism
something like 'probe sha1 sha1_z990 sha1' in modprobe.conf
-> try module sha1_z990, if it fails to load standard module sha1
the 'probe' statement is currently not supported in modprobe.conf
4. Currently implemented z990 crypto algorithms
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
The following crypto algorithms with z990 MSA support are currently implemented.
The name of each algorithm under which it is registered in crypto API and the
name of the respective module is given in square brackets.
- SHA1 Digest Algorithm [sha1 -> sha1_z990]
- DES Encrypt/Decrypt Algorithm (64bit key) [des -> des_z990]
- Triple DES Encrypt/Decrypt Algorithm (128bit key) [des3_ede128 -> des_z990]
- Triple DES Encrypt/Decrypt Algorithm (192bit key) [des3_ede -> des_z990]
In order to load, for example, the sha1_z990 module when the sha1 algorithm is
requested (see 3.2.) add 'alias sha1 sha1_z990' to /etc/modprobe.conf.
s390 SCSI dump tool (zfcpdump)
System z machines (z900 or higher) provide hardware support for creating system
dumps on SCSI disks. The dump process is initiated by booting a dump tool, which
has to create a dump of the current (probably crashed) Linux image. In order to
not overwrite memory of the crashed Linux with data of the dump tool, the
hardware saves some memory plus the register sets of the boot cpu before the
dump tool is loaded. There exists an SCLP hardware interface to obtain the saved
memory afterwards. Currently 32 MB are saved.
This zfcpdump implementation consists of a Linux dump kernel together with
a userspace dump tool, which are loaded together into the saved memory region
below 32 MB. zfcpdump is installed on a SCSI disk using zipl (as contained in
the s390-tools package) to make the device bootable. The operator of a Linux
system can then trigger a SCSI dump by booting the SCSI disk, where zfcpdump
resides on.
The kernel part of zfcpdump is implemented as a debugfs file under "zcore/mem",
which exports memory and registers of the crashed Linux in an s390
standalone dump format. It can be used in the same way as e.g. /dev/mem. The
dump format defines a 4K header followed by plain uncompressed memory. The
register sets are stored in the prefix pages of the respective cpus. To build a
dump enabled kernel with the zcore driver, the kernel config option
CONFIG_ZFCPDUMP has to be set. When reading from "zcore/mem", the part of
memory, which has been saved by hardware is read by the driver via the SCLP
hardware interface. The second part is just copied from the non overwritten real
memory.
The userspace application of zfcpdump can reside e.g. in an intitramfs or an
initrd. It reads from zcore/mem and writes the system dump to a file on a
SCSI disk.
To build a zfcpdump kernel use the following settings in your kernel
configuration:
* CONFIG_ZFCPDUMP=y
* Enable ZFCP driver
* Enable SCSI driver
* Enable ext2 and ext3 filesystems
* Disable as many features as possible to keep the kernel small.
E.g. network support is not needed at all.
To use the zfcpdump userspace application in an initramfs you have to do the
following:
* Copy the zfcpdump executable somewhere into your Linux tree.
E.g. to "arch/s390/boot/zfcpdump. If you do not want to include
shared libraries, compile the tool with the "-static" gcc option.
* If you want to include e2fsck, add it to your source tree, too. The zfcpdump
application attempts to start /sbin/e2fsck from the ramdisk.
* Use an initramfs config file like the following:
dir /dev 755 0 0
nod /dev/console 644 0 0 c 5 1
nod /dev/null 644 0 0 c 1 3
nod /dev/sda1 644 0 0 b 8 1
nod /dev/sda2 644 0 0 b 8 2
nod /dev/sda3 644 0 0 b 8 3
nod /dev/sda4 644 0 0 b 8 4
nod /dev/sda5 644 0 0 b 8 5
nod /dev/sda6 644 0 0 b 8 6
nod /dev/sda7 644 0 0 b 8 7
nod /dev/sda8 644 0 0 b 8 8
nod /dev/sda9 644 0 0 b 8 9
nod /dev/sda10 644 0 0 b 8 10
nod /dev/sda11 644 0 0 b 8 11
nod /dev/sda12 644 0 0 b 8 12
nod /dev/sda13 644 0 0 b 8 13
nod /dev/sda14 644 0 0 b 8 14
nod /dev/sda15 644 0 0 b 8 15
file /init arch/s390/boot/zfcpdump 755 0 0
file /sbin/e2fsck arch/s390/boot/e2fsck 755 0 0
dir /proc 755 0 0
dir /sys 755 0 0
dir /mnt 755 0 0
dir /sbin 755 0 0
* Issue "make image" to build the zfcpdump image with initramfs.
In a Linux distribution the zfcpdump enabled kernel image must be copied to
/usr/share/zfcpdump/zfcpdump.image, where the s390 zipl tool is looking for the
dump kernel when preparing a SCSI dump disk.
If you use a ramdisk copy it to "/usr/share/zfcpdump/zfcpdump.rd".
For more information on how to use zfcpdump refer to the s390 'Using the Dump
Tools book', which is available from
http://www.ibm.com/developerworks/linux/linux390.
...@@ -41,6 +41,11 @@ config GENERIC_HWEIGHT ...@@ -41,6 +41,11 @@ config GENERIC_HWEIGHT
config GENERIC_TIME config GENERIC_TIME
def_bool y def_bool y
config GENERIC_BUG
bool
depends on BUG
default y
config NO_IOMEM config NO_IOMEM
def_bool y def_bool y
...@@ -514,6 +519,14 @@ config KEXEC ...@@ -514,6 +519,14 @@ config KEXEC
current kernel, and to start another kernel. It is like a reboot current kernel, and to start another kernel. It is like a reboot
but is independent of hardware/microcode support. but is independent of hardware/microcode support.
config ZFCPDUMP
tristate "zfcpdump support"
select SMP
default n
help
Select this option if you want to build an zfcpdump enabled kernel.
Refer to "Documentation/s390/zfcpdump.txt" for more details on this.
endmenu endmenu
source "net/Kconfig" source "net/Kconfig"
......
...@@ -67,8 +67,10 @@ endif ...@@ -67,8 +67,10 @@ endif
ifeq ($(call cc-option-yn,-mstack-size=8192 -mstack-guard=128),y) ifeq ($(call cc-option-yn,-mstack-size=8192 -mstack-guard=128),y)
cflags-$(CONFIG_CHECK_STACK) += -mstack-size=$(STACK_SIZE) cflags-$(CONFIG_CHECK_STACK) += -mstack-size=$(STACK_SIZE)
ifneq ($(call cc-option-yn,-mstack-size=8192),y)
cflags-$(CONFIG_CHECK_STACK) += -mstack-guard=$(CONFIG_STACK_GUARD) cflags-$(CONFIG_CHECK_STACK) += -mstack-guard=$(CONFIG_STACK_GUARD)
endif endif
endif
ifeq ($(call cc-option-yn,-mwarn-dynamicstack),y) ifeq ($(call cc-option-yn,-mwarn-dynamicstack),y)
cflags-$(CONFIG_WARN_STACK) += -mwarn-dynamicstack cflags-$(CONFIG_WARN_STACK) += -mwarn-dynamicstack
...@@ -103,6 +105,9 @@ install: vmlinux ...@@ -103,6 +105,9 @@ install: vmlinux
image: vmlinux image: vmlinux
$(Q)$(MAKE) $(build)=$(boot) $(boot)/$@ $(Q)$(MAKE) $(build)=$(boot) $(boot)/$@
zfcpdump:
$(Q)$(MAKE) $(build)=$(boot) $(boot)/$@
archclean: archclean:
$(Q)$(MAKE) $(clean)=$(boot) $(Q)$(MAKE) $(clean)=$(boot)
......
...@@ -668,45 +668,7 @@ EXPORT_SYMBOL_GPL(appldata_register_ops); ...@@ -668,45 +668,7 @@ EXPORT_SYMBOL_GPL(appldata_register_ops);
EXPORT_SYMBOL_GPL(appldata_unregister_ops); EXPORT_SYMBOL_GPL(appldata_unregister_ops);
EXPORT_SYMBOL_GPL(appldata_diag); EXPORT_SYMBOL_GPL(appldata_diag);
#ifdef MODULE
/*
* Kernel symbols needed by appldata_mem and appldata_os modules.
* However, if this file is compiled as a module (for testing only), these
* symbols are not exported. In this case, we define them locally and export
* those.
*/
void si_swapinfo(struct sysinfo *val)
{
val->freeswap = -1ul;
val->totalswap = -1ul;
}
unsigned long avenrun[3] = {-1 - FIXED_1/200, -1 - FIXED_1/200,
-1 - FIXED_1/200};
int nr_threads = -1;
void get_full_page_state(struct page_state *ps)
{
memset(ps, -1, sizeof(struct page_state));
}
unsigned long nr_running(void)
{
return -1;
}
unsigned long nr_iowait(void)
{
return -1;
}
/*unsigned long nr_context_switches(void)
{
return -1;
}*/
#endif /* MODULE */
EXPORT_SYMBOL_GPL(si_swapinfo); EXPORT_SYMBOL_GPL(si_swapinfo);
EXPORT_SYMBOL_GPL(nr_threads); EXPORT_SYMBOL_GPL(nr_threads);
EXPORT_SYMBOL_GPL(nr_running); EXPORT_SYMBOL_GPL(nr_running);
EXPORT_SYMBOL_GPL(nr_iowait); EXPORT_SYMBOL_GPL(nr_iowait);
//EXPORT_SYMBOL_GPL(nr_context_switches);
...@@ -25,99 +25,100 @@ ...@@ -25,99 +25,100 @@
*/ */
#include <linux/init.h> #include <linux/init.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/mm.h>
#include <linux/crypto.h> #include <linux/crypto.h>
#include <asm/scatterlist.h>
#include <asm/byteorder.h>
#include "crypt_s390.h" #include "crypt_s390.h"
#define SHA1_DIGEST_SIZE 20 #define SHA1_DIGEST_SIZE 20
#define SHA1_BLOCK_SIZE 64 #define SHA1_BLOCK_SIZE 64
struct crypt_s390_sha1_ctx { struct s390_sha1_ctx {
u64 count; u64 count; /* message length */
u32 state[5]; u32 state[5];
u32 buf_len; u8 buf[2 * SHA1_BLOCK_SIZE];
u8 buffer[2 * SHA1_BLOCK_SIZE];
}; };
static void sha1_init(struct crypto_tfm *tfm) static void sha1_init(struct crypto_tfm *tfm)
{ {
struct crypt_s390_sha1_ctx *ctx = crypto_tfm_ctx(tfm); struct s390_sha1_ctx *sctx = crypto_tfm_ctx(tfm);
ctx->state[0] = 0x67452301; sctx->state[0] = 0x67452301;
ctx->state[1] = 0xEFCDAB89; sctx->state[1] = 0xEFCDAB89;
ctx->state[2] = 0x98BADCFE; sctx->state[2] = 0x98BADCFE;
ctx->state[3] = 0x10325476; sctx->state[3] = 0x10325476;
ctx->state[4] = 0xC3D2E1F0; sctx->state[4] = 0xC3D2E1F0;
sctx->count = 0;
ctx->count = 0;
ctx->buf_len = 0;
} }
static void sha1_update(struct crypto_tfm *tfm, const u8 *data, static void sha1_update(struct crypto_tfm *tfm, const u8 *data,
unsigned int len) unsigned int len)
{ {
struct crypt_s390_sha1_ctx *sctx; struct s390_sha1_ctx *sctx = crypto_tfm_ctx(tfm);
long imd_len; unsigned int index;
int ret;
sctx = crypto_tfm_ctx(tfm);
sctx->count += len * 8; /* message bit length */ /* how much is already in the buffer? */
index = sctx->count & 0x3f;
/* anything in buffer yet? -> must be completed */
if (sctx->buf_len && (sctx->buf_len + len) >= SHA1_BLOCK_SIZE) { sctx->count += len;
/* complete full block and hash */
memcpy(sctx->buffer + sctx->buf_len, data, if (index + len < SHA1_BLOCK_SIZE)
SHA1_BLOCK_SIZE - sctx->buf_len); goto store;
crypt_s390_kimd(KIMD_SHA_1, sctx->state, sctx->buffer,
/* process one stored block */
if (index) {
memcpy(sctx->buf + index, data, SHA1_BLOCK_SIZE - index);
ret = crypt_s390_kimd(KIMD_SHA_1, sctx->state, sctx->buf,
SHA1_BLOCK_SIZE); SHA1_BLOCK_SIZE);
data += SHA1_BLOCK_SIZE - sctx->buf_len; BUG_ON(ret != SHA1_BLOCK_SIZE);
len -= SHA1_BLOCK_SIZE - sctx->buf_len; data += SHA1_BLOCK_SIZE - index;
sctx->buf_len = 0; len -= SHA1_BLOCK_SIZE - index;
} }
/* rest of data contains full blocks? */ /* process as many blocks as possible */
imd_len = len & ~0x3ful; if (len >= SHA1_BLOCK_SIZE) {
if (imd_len) { ret = crypt_s390_kimd(KIMD_SHA_1, sctx->state, data,
crypt_s390_kimd(KIMD_SHA_1, sctx->state, data, imd_len); len & ~(SHA1_BLOCK_SIZE - 1));
data += imd_len; BUG_ON(ret != (len & ~(SHA1_BLOCK_SIZE - 1)));
len -= imd_len; data += ret;
} len -= ret;
/* anything left? store in buffer */
if (len) {
memcpy(sctx->buffer + sctx->buf_len , data, len);
sctx->buf_len += len;
} }
}
store:
/* anything left? */
if (len)
memcpy(sctx->buf + index , data, len);
}
static void pad_message(struct crypt_s390_sha1_ctx* sctx) /* Add padding and return the message digest. */
static void sha1_final(struct crypto_tfm *tfm, u8 *out)
{ {
int index; struct s390_sha1_ctx *sctx = crypto_tfm_ctx(tfm);
u64 bits;
unsigned int index, end;
int ret;
/* must perform manual padding */
index = sctx->count & 0x3f;
end = (index < 56) ? SHA1_BLOCK_SIZE : (2 * SHA1_BLOCK_SIZE);
index = sctx->buf_len;
sctx->buf_len = (sctx->buf_len < 56) ?
SHA1_BLOCK_SIZE:2 * SHA1_BLOCK_SIZE;
/* start pad with 1 */ /* start pad with 1 */
sctx->buffer[index] = 0x80; sctx->buf[index] = 0x80;
/* pad with zeros */ /* pad with zeros */
index++; index++;
memset(sctx->buffer + index, 0x00, sctx->buf_len - index); memset(sctx->buf + index, 0x00, end - index - 8);
/* append length */
memcpy(sctx->buffer + sctx->buf_len - 8, &sctx->count,
sizeof sctx->count);
}
/* Add padding and return the message digest. */ /* append message length */
static void sha1_final(struct crypto_tfm *tfm, u8 *out) bits = sctx->count * 8;
{ memcpy(sctx->buf + end - 8, &bits, sizeof(bits));
struct crypt_s390_sha1_ctx *sctx = crypto_tfm_ctx(tfm);
ret = crypt_s390_kimd(KIMD_SHA_1, sctx->state, sctx->buf, end);
BUG_ON(ret != end);
/* must perform manual padding */
pad_message(sctx);
crypt_s390_kimd(KIMD_SHA_1, sctx->state, sctx->buffer, sctx->buf_len);
/* copy digest to out */ /* copy digest to out */
memcpy(out, sctx->state, SHA1_DIGEST_SIZE); memcpy(out, sctx->state, SHA1_DIGEST_SIZE);
/* wipe context */ /* wipe context */
memset(sctx, 0, sizeof *sctx); memset(sctx, 0, sizeof *sctx);
} }
...@@ -128,7 +129,7 @@ static struct crypto_alg alg = { ...@@ -128,7 +129,7 @@ static struct crypto_alg alg = {
.cra_priority = CRYPT_S390_PRIORITY, .cra_priority = CRYPT_S390_PRIORITY,
.cra_flags = CRYPTO_ALG_TYPE_DIGEST, .cra_flags = CRYPTO_ALG_TYPE_DIGEST,
.cra_blocksize = SHA1_BLOCK_SIZE, .cra_blocksize = SHA1_BLOCK_SIZE,
.cra_ctxsize = sizeof(struct crypt_s390_sha1_ctx), .cra_ctxsize = sizeof(struct s390_sha1_ctx),
.cra_module = THIS_MODULE, .cra_module = THIS_MODULE,
.cra_list = LIST_HEAD_INIT(alg.cra_list), .cra_list = LIST_HEAD_INIT(alg.cra_list),
.cra_u = { .digest = { .cra_u = { .digest = {
......
...@@ -26,7 +26,7 @@ ...@@ -26,7 +26,7 @@
#define SHA256_BLOCK_SIZE 64 #define SHA256_BLOCK_SIZE 64
struct s390_sha256_ctx { struct s390_sha256_ctx {
u64 count; u64 count; /* message length */
u32 state[8]; u32 state[8];
u8 buf[2 * SHA256_BLOCK_SIZE]; u8 buf[2 * SHA256_BLOCK_SIZE];
}; };
...@@ -54,10 +54,9 @@ static void sha256_update(struct crypto_tfm *tfm, const u8 *data, ...@@ -54,10 +54,9 @@ static void sha256_update(struct crypto_tfm *tfm, const u8 *data,
int ret; int ret;
/* how much is already in the buffer? */ /* how much is already in the buffer? */
index = sctx->count / 8 & 0x3f; index = sctx->count & 0x3f;
/* update message bit length */ sctx->count += len;
sctx->count += len * 8;
if ((index + len) < SHA256_BLOCK_SIZE) if ((index + len) < SHA256_BLOCK_SIZE)
goto store; goto store;
...@@ -87,12 +86,17 @@ static void sha256_update(struct crypto_tfm *tfm, const u8 *data, ...@@ -87,12 +86,17 @@ static void sha256_update(struct crypto_tfm *tfm, const u8 *data,
memcpy(sctx->buf + index , data, len); memcpy(sctx->buf + index , data, len);
} }
static void pad_message(struct s390_sha256_ctx* sctx) /* Add padding and return the message digest */
static void sha256_final(struct crypto_tfm *tfm, u8 *out)
{ {
int index, end; struct s390_sha256_ctx *sctx = crypto_tfm_ctx(tfm);
u64 bits;
unsigned int index, end;
int ret;
index = sctx->count / 8 & 0x3f; /* must perform manual padding */
end = index < 56 ? SHA256_BLOCK_SIZE : 2 * SHA256_BLOCK_SIZE; index = sctx->count & 0x3f;
end = (index < 56) ? SHA256_BLOCK_SIZE : (2 * SHA256_BLOCK_SIZE);
/* start pad with 1 */ /* start pad with 1 */
sctx->buf[index] = 0x80; sctx->buf[index] = 0x80;
...@@ -102,21 +106,11 @@ static void pad_message(struct s390_sha256_ctx* sctx) ...@@ -102,21 +106,11 @@ static void pad_message(struct s390_sha256_ctx* sctx)
memset(sctx->buf + index, 0x00, end - index - 8); memset(sctx->buf + index, 0x00, end - index - 8);
/* append message length */ /* append message length */
memcpy(sctx->buf + end - 8, &sctx->count, sizeof sctx->count); bits = sctx->count * 8;
memcpy(sctx->buf + end - 8, &bits, sizeof(bits));
sctx->count = end * 8;
}
/* Add padding and return the message digest */
static void sha256_final(struct crypto_tfm *tfm, u8 *out)
{
struct s390_sha256_ctx *sctx = crypto_tfm_ctx(tfm);
/* must perform manual padding */
pad_message(sctx);
crypt_s390_kimd(KIMD_SHA_256, sctx->state, sctx->buf, ret = crypt_s390_kimd(KIMD_SHA_256, sctx->state, sctx->buf, end);
sctx->count / 8); BUG_ON(ret != end);
/* copy digest to out */ /* copy digest to out */
memcpy(out, sctx->state, SHA256_DIGEST_SIZE); memcpy(out, sctx->state, SHA256_DIGEST_SIZE);
......
...@@ -12,6 +12,7 @@ CONFIG_RWSEM_XCHGADD_ALGORITHM=y ...@@ -12,6 +12,7 @@ CONFIG_RWSEM_XCHGADD_ALGORITHM=y
# CONFIG_ARCH_HAS_ILOG2_U64 is not set # CONFIG_ARCH_HAS_ILOG2_U64 is not set
CONFIG_GENERIC_HWEIGHT=y CONFIG_GENERIC_HWEIGHT=y
CONFIG_GENERIC_TIME=y CONFIG_GENERIC_TIME=y
CONFIG_GENERIC_BUG=y
CONFIG_NO_IOMEM=y CONFIG_NO_IOMEM=y
CONFIG_S390=y CONFIG_S390=y
CONFIG_DEFCONFIG_LIST="/lib/modules/$UNAME_RELEASE/.config" CONFIG_DEFCONFIG_LIST="/lib/modules/$UNAME_RELEASE/.config"
...@@ -166,6 +167,7 @@ CONFIG_NO_IDLE_HZ=y ...@@ -166,6 +167,7 @@ CONFIG_NO_IDLE_HZ=y
CONFIG_NO_IDLE_HZ_INIT=y CONFIG_NO_IDLE_HZ_INIT=y
CONFIG_S390_HYPFS_FS=y CONFIG_S390_HYPFS_FS=y
CONFIG_KEXEC=y CONFIG_KEXEC=y
# CONFIG_ZFCPDUMP is not set
# #
# Networking # Networking
...@@ -705,6 +707,7 @@ CONFIG_DEBUG_MUTEXES=y ...@@ -705,6 +707,7 @@ CONFIG_DEBUG_MUTEXES=y
CONFIG_DEBUG_SPINLOCK_SLEEP=y CONFIG_DEBUG_SPINLOCK_SLEEP=y
# CONFIG_DEBUG_LOCKING_API_SELFTESTS is not set # CONFIG_DEBUG_LOCKING_API_SELFTESTS is not set
# CONFIG_DEBUG_KOBJECT is not set # CONFIG_DEBUG_KOBJECT is not set
CONFIG_DEBUG_BUGVERBOSE=y
# CONFIG_DEBUG_INFO is not set # CONFIG_DEBUG_INFO is not set
# CONFIG_DEBUG_VM is not set # CONFIG_DEBUG_VM is not set
# CONFIG_DEBUG_LIST is not set # CONFIG_DEBUG_LIST is not set
......
...@@ -6,7 +6,7 @@ EXTRA_AFLAGS := -traditional ...@@ -6,7 +6,7 @@ EXTRA_AFLAGS := -traditional
obj-y := bitmap.o traps.o time.o process.o base.o early.o \ obj-y := bitmap.o traps.o time.o process.o base.o early.o \
setup.o sys_s390.o ptrace.o signal.o cpcmd.o ebcdic.o \ setup.o sys_s390.o ptrace.o signal.o cpcmd.o ebcdic.o \
semaphore.o s390_ext.o debug.o irq.o ipl.o semaphore.o s390_ext.o debug.o irq.o ipl.o dis.o
obj-y += $(if $(CONFIG_64BIT),entry64.o,entry.o) obj-y += $(if $(CONFIG_64BIT),entry64.o,entry.o)
obj-y += $(if $(CONFIG_64BIT),reipl64.o,reipl.o) obj-y += $(if $(CONFIG_64BIT),reipl64.o,reipl.o)
......
...@@ -495,29 +495,34 @@ sys32_rt_sigqueueinfo(int pid, int sig, compat_siginfo_t __user *uinfo) ...@@ -495,29 +495,34 @@ sys32_rt_sigqueueinfo(int pid, int sig, compat_siginfo_t __user *uinfo)
* sys32_execve() executes a new program after the asm stub has set * sys32_execve() executes a new program after the asm stub has set
* things up for us. This should basically do what I want it to. * things up for us. This should basically do what I want it to.
*/ */
asmlinkage long asmlinkage long sys32_execve(void)
sys32_execve(struct pt_regs regs)
{ {
int error; struct pt_regs *regs = task_pt_regs(current);
char * filename; char *filename;
unsigned long result;
int rc;
filename = getname(compat_ptr(regs.orig_gpr2)); filename = getname(compat_ptr(regs->orig_gpr2));
error = PTR_ERR(filename); if (IS_ERR(filename)) {
if (IS_ERR(filename)) result = PTR_ERR(filename);
goto out; goto out;
error = compat_do_execve(filename, compat_ptr(regs.gprs[3]), }
compat_ptr(regs.gprs[4]), &regs); rc = compat_do_execve(filename, compat_ptr(regs->gprs[3]),
if (error == 0) compat_ptr(regs->gprs[4]), regs);
{ if (rc) {
result = rc;
goto out_putname;
}
task_lock(current); task_lock(current);
current->ptrace &= ~PT_DTRACE; current->ptrace &= ~PT_DTRACE;
task_unlock(current); task_unlock(current);
current->thread.fp_regs.fpc=0; current->thread.fp_regs.fpc=0;
asm volatile("sfpc %0,0" : : "d" (0)); asm volatile("sfpc %0,0" : : "d" (0));
} result = regs->gprs[2];
out_putname:
putname(filename); putname(filename);
out: out:
return error; return result;
} }
...@@ -918,19 +923,20 @@ asmlinkage long sys32_write(unsigned int fd, char __user * buf, size_t count) ...@@ -918,19 +923,20 @@ asmlinkage long sys32_write(unsigned int fd, char __user * buf, size_t count)
return sys_write(fd, buf, count); return sys_write(fd, buf, count);
} }
asmlinkage long sys32_clone(struct pt_regs regs) asmlinkage long sys32_clone(void)
{ {
struct pt_regs *regs = task_pt_regs(current);
unsigned long clone_flags; unsigned long clone_flags;
unsigned long newsp; unsigned long newsp;
int __user *parent_tidptr, *child_tidptr; int __user *parent_tidptr, *child_tidptr;
clone_flags = regs.gprs[3] & 0xffffffffUL; clone_flags = regs->gprs[3] & 0xffffffffUL;
newsp = regs.orig_gpr2 & 0x7fffffffUL; newsp = regs->orig_gpr2 & 0x7fffffffUL;
parent_tidptr = compat_ptr(regs.gprs[4]); parent_tidptr = compat_ptr(regs->gprs[4]);
child_tidptr = compat_ptr(regs.gprs[5]); child_tidptr = compat_ptr(regs->gprs[5]);
if (!newsp) if (!newsp)
newsp = regs.gprs[15]; newsp = regs->gprs[15];
return do_fork(clone_flags, newsp, &regs, 0, return do_fork(clone_flags, newsp, regs, 0,
parent_tidptr, child_tidptr); parent_tidptr, child_tidptr);
} }
......
...@@ -255,9 +255,9 @@ sys32_rt_sigaction(int sig, const struct sigaction32 __user *act, ...@@ -255,9 +255,9 @@ sys32_rt_sigaction(int sig, const struct sigaction32 __user *act,
} }
asmlinkage long asmlinkage long
sys32_sigaltstack(const stack_t32 __user *uss, stack_t32 __user *uoss, sys32_sigaltstack(const stack_t32 __user *uss, stack_t32 __user *uoss)
struct pt_regs *regs)
{ {
struct pt_regs *regs = task_pt_regs(current);
stack_t kss, koss; stack_t kss, koss;
unsigned long ss_sp; unsigned long ss_sp;
int ret, err = 0; int ret, err = 0;
...@@ -344,8 +344,9 @@ static int restore_sigregs32(struct pt_regs *regs,_sigregs32 __user *sregs) ...@@ -344,8 +344,9 @@ static int restore_sigregs32(struct pt_regs *regs,_sigregs32 __user *sregs)
return 0; return 0;
} }
asmlinkage long sys32_sigreturn(struct pt_regs *regs) asmlinkage long sys32_sigreturn(void)
{ {
struct pt_regs *regs = task_pt_regs(current);
sigframe32 __user *frame = (sigframe32 __user *)regs->gprs[15]; sigframe32 __user *frame = (sigframe32 __user *)regs->gprs[15];
sigset_t set; sigset_t set;
...@@ -370,8 +371,9 @@ asmlinkage long sys32_sigreturn(struct pt_regs *regs) ...@@ -370,8 +371,9 @@ asmlinkage long sys32_sigreturn(struct pt_regs *regs)
return 0; return 0;
} }
asmlinkage long sys32_rt_sigreturn(struct pt_regs *regs) asmlinkage long sys32_rt_sigreturn(void)
{ {
struct pt_regs *regs = task_pt_regs(current);
rt_sigframe32 __user *frame = (rt_sigframe32 __user *)regs->gprs[15]; rt_sigframe32 __user *frame = (rt_sigframe32 __user *)regs->gprs[15];
sigset_t set; sigset_t set;
stack_t st; stack_t st;
......
This diff is collapsed.
...@@ -253,11 +253,10 @@ static noinline __init void find_memory_chunks(unsigned long memsize) ...@@ -253,11 +253,10 @@ static noinline __init void find_memory_chunks(unsigned long memsize)
break; break;
#endif #endif
/* /*
* Finish memory detection at the first hole, unless * Finish memory detection at the first hole
* - we reached the hsa -> skip it. * if storage size is unknown.
* - we know there must be more.
*/ */
if (cc == -1UL && !memsize && old_addr != ADDR2G) if (cc == -1UL && !memsize)
break; break;
if (memsize && addr >= memsize) if (memsize && addr >= memsize)
break; break;
......
...@@ -249,8 +249,6 @@ sysc_do_restart: ...@@ -249,8 +249,6 @@ sysc_do_restart:
bnz BASED(sysc_tracesys) bnz BASED(sysc_tracesys)
basr %r14,%r8 # call sys_xxxx basr %r14,%r8 # call sys_xxxx
st %r2,SP_R2(%r15) # store return value (change R2 on stack) st %r2,SP_R2(%r15) # store return value (change R2 on stack)
# ATTENTION: check sys_execve_glue before
# changing anything here !!
sysc_return: sysc_return:
tm SP_PSW+1(%r15),0x01 # returning to user ? tm SP_PSW+1(%r15),0x01 # returning to user ?
...@@ -381,50 +379,37 @@ ret_from_fork: ...@@ -381,50 +379,37 @@ ret_from_fork:
b BASED(sysc_return) b BASED(sysc_return)
# #
# clone, fork, vfork, exec and sigreturn need glue, # kernel_execve function needs to deal with pt_regs that is not
# because they all expect pt_regs as parameter, # at the usual place
# but are called with different parameter.
# return-address is set up above
# #
sys_clone_glue: .globl kernel_execve
la %r2,SP_PTREGS(%r15) # load pt_regs kernel_execve:
l %r1,BASED(.Lclone) stm %r12,%r15,48(%r15)
br %r1 # branch to sys_clone lr %r14,%r15
l %r13,__LC_SVC_NEW_PSW+4
sys_fork_glue: s %r15,BASED(.Lc_spsize)
la %r2,SP_PTREGS(%r15) # load pt_regs st %r14,__SF_BACKCHAIN(%r15)
l %r1,BASED(.Lfork) la %r12,SP_PTREGS(%r15)
br %r1 # branch to sys_fork xc 0(__PT_SIZE,%r12),0(%r12)
l %r1,BASED(.Ldo_execve)
sys_vfork_glue: lr %r5,%r12
la %r2,SP_PTREGS(%r15) # load pt_regs basr %r14,%r1
l %r1,BASED(.Lvfork) ltr %r2,%r2
br %r1 # branch to sys_vfork be BASED(0f)
a %r15,BASED(.Lc_spsize)
sys_execve_glue: lm %r12,%r15,48(%r15)
la %r2,SP_PTREGS(%r15) # load pt_regs br %r14
l %r1,BASED(.Lexecve) # execve succeeded.
lr %r12,%r14 # save return address 0: stnsm __SF_EMPTY(%r15),0xfc # disable interrupts
basr %r14,%r1 # call sys_execve l %r15,__LC_KERNEL_STACK # load ksp
ltr %r2,%r2 # check if execve failed s %r15,BASED(.Lc_spsize) # make room for registers & psw
bnz 0(%r12) # it did fail -> store result in gpr2 l %r9,__LC_THREAD_INFO
b 4(%r12) # SKIP ST 2,SP_R2(15) after BASR 14,8 mvc SP_PTREGS(__PT_SIZE,%r15),0(%r12) # copy pt_regs
# in system_call/sysc_tracesys xc __SF_BACKCHAIN(4,%r15),__SF_BACKCHAIN(%r15)
stosm __SF_EMPTY(%r15),0x03 # reenable interrupts
sys_sigreturn_glue: l %r1,BASED(.Lexecve_tail)
la %r2,SP_PTREGS(%r15) # load pt_regs as parameter basr %r14,%r1
l %r1,BASED(.Lsigreturn) b BASED(sysc_return)
br %r1 # branch to sys_sigreturn
sys_rt_sigreturn_glue:
la %r2,SP_PTREGS(%r15) # load pt_regs as parameter
l %r1,BASED(.Lrt_sigreturn)
br %r1 # branch to sys_sigreturn
sys_sigaltstack_glue:
la %r4,SP_PTREGS(%r15) # load pt_regs as parameter
l %r1,BASED(.Lsigaltstack)
br %r1 # branch to sys_sigreturn
/* /*
* Program check handler routine * Program check handler routine
...@@ -1031,19 +1016,11 @@ cleanup_io_leave_insn: ...@@ -1031,19 +1016,11 @@ cleanup_io_leave_insn:
.Ldo_extint: .long do_extint .Ldo_extint: .long do_extint
.Ldo_signal: .long do_signal .Ldo_signal: .long do_signal
.Lhandle_per: .long do_single_step .Lhandle_per: .long do_single_step
.Ldo_execve: .long do_execve
.Lexecve_tail: .long execve_tail
.Ljump_table: .long pgm_check_table .Ljump_table: .long pgm_check_table
.Lschedule: .long schedule .Lschedule: .long schedule
.Lclone: .long sys_clone
.Lexecve: .long sys_execve
.Lfork: .long sys_fork
.Lrt_sigreturn: .long sys_rt_sigreturn
.Lrt_sigsuspend:
.long sys_rt_sigsuspend
.Lsigreturn: .long sys_sigreturn
.Lsigsuspend: .long sys_sigsuspend
.Lsigaltstack: .long sys_sigaltstack
.Ltrace: .long syscall_trace .Ltrace: .long syscall_trace
.Lvfork: .long sys_vfork
.Lschedtail: .long schedule_tail .Lschedtail: .long schedule_tail
.Lsysc_table: .long sys_call_table .Lsysc_table: .long sys_call_table
#ifdef CONFIG_TRACE_IRQFLAGS #ifdef CONFIG_TRACE_IRQFLAGS
......
...@@ -244,8 +244,6 @@ sysc_noemu: ...@@ -244,8 +244,6 @@ sysc_noemu:
jnz sysc_tracesys jnz sysc_tracesys
basr %r14,%r8 # call sys_xxxx basr %r14,%r8 # call sys_xxxx
stg %r2,SP_R2(%r15) # store return value (change R2 on stack) stg %r2,SP_R2(%r15) # store return value (change R2 on stack)
# ATTENTION: check sys_execve_glue before
# changing anything here !!
sysc_return: sysc_return:
tm SP_PSW+1(%r15),0x01 # returning to user ? tm SP_PSW+1(%r15),0x01 # returning to user ?
...@@ -371,77 +369,35 @@ ret_from_fork: ...@@ -371,77 +369,35 @@ ret_from_fork:
j sysc_return j sysc_return
# #
# clone, fork, vfork, exec and sigreturn need glue, # kernel_execve function needs to deal with pt_regs that is not
# because they all expect pt_regs as parameter, # at the usual place
# but are called with different parameter.
# return-address is set up above
# #
sys_clone_glue: .globl kernel_execve
la %r2,SP_PTREGS(%r15) # load pt_regs kernel_execve:
jg sys_clone # branch to sys_clone stmg %r12,%r15,96(%r15)
lgr %r14,%r15
#ifdef CONFIG_COMPAT aghi %r15,-SP_SIZE
sys32_clone_glue: stg %r14,__SF_BACKCHAIN(%r15)
la %r2,SP_PTREGS(%r15) # load pt_regs la %r12,SP_PTREGS(%r15)
jg sys32_clone # branch to sys32_clone xc 0(__PT_SIZE,%r12),0(%r12)
#endif lgr %r5,%r12
brasl %r14,do_execve
sys_fork_glue: ltgfr %r2,%r2
la %r2,SP_PTREGS(%r15) # load pt_regs je 0f
jg sys_fork # branch to sys_fork aghi %r15,SP_SIZE
lmg %r12,%r15,96(%r15)
sys_vfork_glue: br %r14
la %r2,SP_PTREGS(%r15) # load pt_regs # execve succeeded.
jg sys_vfork # branch to sys_vfork 0: stnsm __SF_EMPTY(%r15),0xfc # disable interrupts
lg %r15,__LC_KERNEL_STACK # load ksp
sys_execve_glue: aghi %r15,-SP_SIZE # make room for registers & psw
la %r2,SP_PTREGS(%r15) # load pt_regs lg %r13,__LC_SVC_NEW_PSW+8
lgr %r12,%r14 # save return address lg %r9,__LC_THREAD_INFO
brasl %r14,sys_execve # call sys_execve mvc SP_PTREGS(__PT_SIZE,%r15),0(%r12) # copy pt_regs
ltgr %r2,%r2 # check if execve failed xc __SF_BACKCHAIN(8,%r15),__SF_BACKCHAIN(%r15)
bnz 0(%r12) # it did fail -> store result in gpr2 stosm __SF_EMPTY(%r15),0x03 # reenable interrupts
b 6(%r12) # SKIP STG 2,SP_R2(15) in brasl %r14,execve_tail
# system_call/sysc_tracesys j sysc_return
#ifdef CONFIG_COMPAT
sys32_execve_glue:
la %r2,SP_PTREGS(%r15) # load pt_regs
lgr %r12,%r14 # save return address
brasl %r14,sys32_execve # call sys32_execve
ltgr %r2,%r2 # check if execve failed
bnz 0(%r12) # it did fail -> store result in gpr2
b 6(%r12) # SKIP STG 2,SP_R2(15) in
# system_call/sysc_tracesys
#endif
sys_sigreturn_glue:
la %r2,SP_PTREGS(%r15) # load pt_regs as parameter
jg sys_sigreturn # branch to sys_sigreturn
#ifdef CONFIG_COMPAT
sys32_sigreturn_glue:
la %r2,SP_PTREGS(%r15) # load pt_regs as parameter
jg sys32_sigreturn # branch to sys32_sigreturn
#endif
sys_rt_sigreturn_glue:
la %r2,SP_PTREGS(%r15) # load pt_regs as parameter
jg sys_rt_sigreturn # branch to sys_sigreturn
#ifdef CONFIG_COMPAT
sys32_rt_sigreturn_glue:
la %r2,SP_PTREGS(%r15) # load pt_regs as parameter
jg sys32_rt_sigreturn # branch to sys32_sigreturn
#endif
sys_sigaltstack_glue:
la %r4,SP_PTREGS(%r15) # load pt_regs as parameter
jg sys_sigaltstack # branch to sys_sigreturn
#ifdef CONFIG_COMPAT
sys32_sigaltstack_glue:
la %r4,SP_PTREGS(%r15) # load pt_regs as parameter
jg sys32_sigaltstack_wrapper # branch to sys_sigreturn
#endif
/* /*
* Program check handler routine * Program check handler routine
......
...@@ -39,7 +39,69 @@ startup_continue: ...@@ -39,7 +39,69 @@ startup_continue:
basr %r13,0 # get base basr %r13,0 # get base
.LPG1: sll %r13,1 # remove high order bit .LPG1: sll %r13,1 # remove high order bit
srl %r13,1 srl %r13,1
lhi %r1,1 # mode 1 = esame
#ifdef CONFIG_ZFCPDUMP
# check if we have been ipled using zfcp dump:
tm 0xb9,0x01 # test if subchannel is enabled
jno .nodump # subchannel disabled
l %r1,0xb8
la %r5,.Lipl_schib-.LPG1(%r13)
stsch 0(%r5) # get schib of subchannel
jne .nodump # schib not available
tm 5(%r5),0x01 # devno valid?
jno .nodump
tm 4(%r5),0x80 # qdio capable device?
jno .nodump
l %r2,20(%r0) # address of ipl parameter block
lhi %r3,0
ic %r3,0x148(%r2) # get opt field
chi %r3,0x20 # load with dump?
jne .nodump
# store all prefix registers in case of load with dump:
la %r7,0 # base register for 0 page
la %r8,0 # first cpu
l %r11,.Lpref_arr_ptr-.LPG1(%r13) # address of prefix array
ahi %r11,4 # skip boot cpu
lr %r12,%r11
ahi %r12,(CONFIG_NR_CPUS*4) # end of prefix array
stap .Lcurrent_cpu+2-.LPG1(%r13) # store current cpu addr
1:
cl %r8,.Lcurrent_cpu-.LPG1(%r13) # is ipl cpu ?
je 4f # if yes get next cpu
2:
lr %r9,%r7
sigp %r9,%r8,0x9 # stop & store status of cpu
brc 8,3f # accepted
brc 4,4f # status stored: next cpu
brc 2,2b # busy: try again
brc 1,4f # not op: next cpu
3:
mvc 0(4,%r11),264(%r7) # copy prefix register to prefix array
ahi %r11,4 # next element in prefix array
clr %r11,%r12
je 5f # no more space in prefix array
4:
ahi %r8,1 # next cpu (r8 += 1)
cl %r8,.Llast_cpu-.LPG1(%r13) # is last possible cpu ?
jl 1b # jump if not last cpu
5:
lhi %r1,2 # mode 2 = esame (dump)
j 6f
.align 4
.Lipl_schib:
.rept 13
.long 0
.endr
.nodump:
lhi %r1,1 # mode 1 = esame (normal ipl)
6:
#else
lhi %r1,1 # mode 1 = esame (normal ipl)
#endif /* CONFIG_ZFCPDUMP */
mvi __LC_AR_MODE_ID,1 # set esame flag mvi __LC_AR_MODE_ID,1 # set esame flag
slr %r0,%r0 # set cpuid to zero slr %r0,%r0 # set cpuid to zero
sigp %r1,%r0,0x12 # switch to esame mode sigp %r1,%r0,0x12 # switch to esame mode
...@@ -149,6 +211,14 @@ startup_continue: ...@@ -149,6 +211,14 @@ startup_continue:
.L4malign:.quad 0xffffffffffc00000 .L4malign:.quad 0xffffffffffc00000
.Lscan2g:.quad 0x80000000 + 0x20000 - 8 # 2GB + 128K - 8 .Lscan2g:.quad 0x80000000 + 0x20000 - 8 # 2GB + 128K - 8
.Lnop: .long 0x07000700 .Lnop: .long 0x07000700
#ifdef CONFIG_ZFCPDUMP
.Lcurrent_cpu:
.long 0x0
.Llast_cpu:
.long 0x0000ffff
.Lpref_arr_ptr:
.long zfcpdump_prefix_array
#endif /* CONFIG_ZFCPDUMP */
.Lparmaddr: .Lparmaddr:
.quad PARMAREA .quad PARMAREA
.align 64 .align 64
......
This diff is collapsed.
...@@ -31,6 +31,7 @@ ...@@ -31,6 +31,7 @@
#include <linux/string.h> #include <linux/string.h>
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/moduleloader.h> #include <linux/moduleloader.h>
#include <linux/bug.h>
#if 0 #if 0
#define DEBUGP printk #define DEBUGP printk
...@@ -398,9 +399,10 @@ int module_finalize(const Elf_Ehdr *hdr, ...@@ -398,9 +399,10 @@ int module_finalize(const Elf_Ehdr *hdr,
struct module *me) struct module *me)
{ {
vfree(me->arch.syminfo); vfree(me->arch.syminfo);
return 0; return module_bug_finalize(hdr, sechdrs, me);
} }
void module_arch_cleanup(struct module *mod) void module_arch_cleanup(struct module *mod)
{ {
module_bug_cleanup(mod);
} }
...@@ -280,24 +280,26 @@ int copy_thread(int nr, unsigned long clone_flags, unsigned long new_stackp, ...@@ -280,24 +280,26 @@ int copy_thread(int nr, unsigned long clone_flags, unsigned long new_stackp,
return 0; return 0;
} }
asmlinkage long sys_fork(struct pt_regs regs) asmlinkage long sys_fork(void)
{ {
return do_fork(SIGCHLD, regs.gprs[15], &regs, 0, NULL, NULL); struct pt_regs *regs = task_pt_regs(current);
return do_fork(SIGCHLD, regs->gprs[15], regs, 0, NULL, NULL);
} }
asmlinkage long sys_clone(struct pt_regs regs) asmlinkage long sys_clone(void)
{ {
struct pt_regs *regs = task_pt_regs(current);
unsigned long clone_flags; unsigned long clone_flags;
unsigned long newsp; unsigned long newsp;
int __user *parent_tidptr, *child_tidptr; int __user *parent_tidptr, *child_tidptr;
clone_flags = regs.gprs[3]; clone_flags = regs->gprs[3];
newsp = regs.orig_gpr2; newsp = regs->orig_gpr2;
parent_tidptr = (int __user *) regs.gprs[4]; parent_tidptr = (int __user *) regs->gprs[4];
child_tidptr = (int __user *) regs.gprs[5]; child_tidptr = (int __user *) regs->gprs[5];
if (!newsp) if (!newsp)
newsp = regs.gprs[15]; newsp = regs->gprs[15];
return do_fork(clone_flags, newsp, &regs, 0, return do_fork(clone_flags, newsp, regs, 0,
parent_tidptr, child_tidptr); parent_tidptr, child_tidptr);
} }
...@@ -311,40 +313,52 @@ asmlinkage long sys_clone(struct pt_regs regs) ...@@ -311,40 +313,52 @@ asmlinkage long sys_clone(struct pt_regs regs)
* do not have enough call-clobbered registers to hold all * do not have enough call-clobbered registers to hold all
* the information you need. * the information you need.
*/ */
asmlinkage long sys_vfork(struct pt_regs regs) asmlinkage long sys_vfork(void)
{ {
struct pt_regs *regs = task_pt_regs(current);
return do_fork(CLONE_VFORK | CLONE_VM | SIGCHLD, return do_fork(CLONE_VFORK | CLONE_VM | SIGCHLD,
regs.gprs[15], &regs, 0, NULL, NULL); regs->gprs[15], regs, 0, NULL, NULL);
} }
/* asmlinkage void execve_tail(void)
* sys_execve() executes a new program.
*/
asmlinkage long sys_execve(struct pt_regs regs)
{ {
int error;
char * filename;
filename = getname((char __user *) regs.orig_gpr2);
error = PTR_ERR(filename);
if (IS_ERR(filename))
goto out;
error = do_execve(filename, (char __user * __user *) regs.gprs[3],
(char __user * __user *) regs.gprs[4], &regs);
if (error == 0) {
task_lock(current); task_lock(current);
current->ptrace &= ~PT_DTRACE; current->ptrace &= ~PT_DTRACE;
task_unlock(current); task_unlock(current);
current->thread.fp_regs.fpc = 0; current->thread.fp_regs.fpc = 0;
if (MACHINE_HAS_IEEE) if (MACHINE_HAS_IEEE)
asm volatile("sfpc %0,%0" : : "d" (0)); asm volatile("sfpc %0,%0" : : "d" (0));
}
/*
* sys_execve() executes a new program.
*/
asmlinkage long sys_execve(void)
{
struct pt_regs *regs = task_pt_regs(current);
char *filename;
unsigned long result;
int rc;
filename = getname((char __user *) regs->orig_gpr2);
if (IS_ERR(filename)) {
result = PTR_ERR(filename);
goto out;
}
rc = do_execve(filename, (char __user * __user *) regs->gprs[3],
(char __user * __user *) regs->gprs[4], regs);
if (rc) {
result = rc;
goto out_putname;
} }
execve_tail();
result = regs->gprs[2];
out_putname:
putname(filename); putname(filename);
out: out:
return error; return result;
} }
/* /*
* fill in the FPU structure for a core dump. * fill in the FPU structure for a core dump.
*/ */
......
...@@ -285,6 +285,26 @@ static void __init conmode_default(void) ...@@ -285,6 +285,26 @@ static void __init conmode_default(void)
} }
} }
#if defined(CONFIG_ZFCPDUMP) || defined(CONFIG_ZFCPDUMP_MODULE)
static void __init setup_zfcpdump(unsigned int console_devno)
{
static char str[64];
if (ipl_info.type != IPL_TYPE_FCP_DUMP)
return;
if (console_devno != -1)
sprintf(str, "cio_ignore=all,!0.0.%04x,!0.0.%04x",
ipl_info.data.fcp.dev_id.devno, console_devno);
else
sprintf(str, "cio_ignore=all,!0.0.%04x",
ipl_info.data.fcp.dev_id.devno);
strcat(COMMAND_LINE, str);
console_loglevel = 2;
}
#else
static inline void setup_zfcpdump(unsigned int console_devno) {}
#endif /* CONFIG_ZFCPDUMP */
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
void (*_machine_restart)(char *command) = machine_restart_smp; void (*_machine_restart)(char *command) = machine_restart_smp;
void (*_machine_halt)(void) = machine_halt_smp; void (*_machine_halt)(void) = machine_halt_smp;
...@@ -586,13 +606,20 @@ setup_resources(void) ...@@ -586,13 +606,20 @@ setup_resources(void)
} }
} }
unsigned long real_memory_size;
EXPORT_SYMBOL_GPL(real_memory_size);
static void __init setup_memory_end(void) static void __init setup_memory_end(void)
{ {
unsigned long real_size, memory_size; unsigned long memory_size;
unsigned long max_mem, max_phys; unsigned long max_mem, max_phys;
int i; int i;
memory_size = real_size = 0; #if defined(CONFIG_ZFCPDUMP) || defined(CONFIG_ZFCPDUMP_MODULE)
if (ipl_info.type == IPL_TYPE_FCP_DUMP)
memory_end = ZFCPDUMP_HSA_SIZE;
#endif
memory_size = 0;
max_phys = VMALLOC_END_INIT - VMALLOC_MIN_SIZE; max_phys = VMALLOC_END_INIT - VMALLOC_MIN_SIZE;
memory_end &= PAGE_MASK; memory_end &= PAGE_MASK;
...@@ -601,7 +628,8 @@ static void __init setup_memory_end(void) ...@@ -601,7 +628,8 @@ static void __init setup_memory_end(void)
for (i = 0; i < MEMORY_CHUNKS; i++) { for (i = 0; i < MEMORY_CHUNKS; i++) {
struct mem_chunk *chunk = &memory_chunk[i]; struct mem_chunk *chunk = &memory_chunk[i];
real_size = max(real_size, chunk->addr + chunk->size); real_memory_size = max(real_memory_size,
chunk->addr + chunk->size);
if (chunk->addr >= max_mem) { if (chunk->addr >= max_mem) {
memset(chunk, 0, sizeof(*chunk)); memset(chunk, 0, sizeof(*chunk));
continue; continue;
...@@ -765,6 +793,7 @@ setup_arch(char **cmdline_p) ...@@ -765,6 +793,7 @@ setup_arch(char **cmdline_p)
parse_early_param(); parse_early_param();
setup_ipl_info();
setup_memory_end(); setup_memory_end();
setup_addressing_mode(); setup_addressing_mode();
setup_memory(); setup_memory();
...@@ -782,6 +811,9 @@ setup_arch(char **cmdline_p) ...@@ -782,6 +811,9 @@ setup_arch(char **cmdline_p)
/* Setup default console */ /* Setup default console */
conmode_default(); conmode_default();
/* Setup zfcpdump support */
setup_zfcpdump(console_devno);
} }
void print_cpu_info(struct cpuinfo_S390 *cpuinfo) void print_cpu_info(struct cpuinfo_S390 *cpuinfo)
......
...@@ -102,9 +102,9 @@ sys_sigaction(int sig, const struct old_sigaction __user *act, ...@@ -102,9 +102,9 @@ sys_sigaction(int sig, const struct old_sigaction __user *act,
} }
asmlinkage long asmlinkage long
sys_sigaltstack(const stack_t __user *uss, stack_t __user *uoss, sys_sigaltstack(const stack_t __user *uss, stack_t __user *uoss)
struct pt_regs *regs)
{ {
struct pt_regs *regs = task_pt_regs(current);
return do_sigaltstack(uss, uoss, regs->gprs[15]); return do_sigaltstack(uss, uoss, regs->gprs[15]);
} }
...@@ -163,8 +163,9 @@ static int restore_sigregs(struct pt_regs *regs, _sigregs __user *sregs) ...@@ -163,8 +163,9 @@ static int restore_sigregs(struct pt_regs *regs, _sigregs __user *sregs)
return 0; return 0;
} }
asmlinkage long sys_sigreturn(struct pt_regs *regs) asmlinkage long sys_sigreturn(void)
{ {
struct pt_regs *regs = task_pt_regs(current);
sigframe __user *frame = (sigframe __user *)regs->gprs[15]; sigframe __user *frame = (sigframe __user *)regs->gprs[15];
sigset_t set; sigset_t set;
...@@ -189,8 +190,9 @@ asmlinkage long sys_sigreturn(struct pt_regs *regs) ...@@ -189,8 +190,9 @@ asmlinkage long sys_sigreturn(struct pt_regs *regs)
return 0; return 0;
} }
asmlinkage long sys_rt_sigreturn(struct pt_regs *regs) asmlinkage long sys_rt_sigreturn(void)
{ {
struct pt_regs *regs = task_pt_regs(current);
rt_sigframe __user *frame = (rt_sigframe __user *)regs->gprs[15]; rt_sigframe __user *frame = (rt_sigframe __user *)regs->gprs[15];
sigset_t set; sigset_t set;
......
This diff is collapsed.
...@@ -266,23 +266,3 @@ s390_fadvise64_64(struct fadvise64_64_args __user *args) ...@@ -266,23 +266,3 @@ s390_fadvise64_64(struct fadvise64_64_args __user *args)
return -EFAULT; return -EFAULT;
return sys_fadvise64_64(a.fd, a.offset, a.len, a.advice); return sys_fadvise64_64(a.fd, a.offset, a.len, a.advice);
} }
/*
* Do a system call from kernel instead of calling sys_execve so we
* end up with proper pt_regs.
*/
int kernel_execve(const char *filename, char *const argv[], char *const envp[])
{
register const char *__arg1 asm("2") = filename;
register char *const*__arg2 asm("3") = argv;
register char *const*__arg3 asm("4") = envp;
register long __svcres asm("2");
asm volatile(
"svc %b1"
: "=d" (__svcres)
: "i" (__NR_execve),
"0" (__arg1),
"d" (__arg2),
"d" (__arg3) : "memory");
return __svcres;
}
...@@ -10,7 +10,7 @@ ...@@ -10,7 +10,7 @@
NI_SYSCALL /* 0 */ NI_SYSCALL /* 0 */
SYSCALL(sys_exit,sys_exit,sys32_exit_wrapper) SYSCALL(sys_exit,sys_exit,sys32_exit_wrapper)
SYSCALL(sys_fork_glue,sys_fork_glue,sys_fork_glue) SYSCALL(sys_fork,sys_fork,sys_fork)
SYSCALL(sys_read,sys_read,sys32_read_wrapper) SYSCALL(sys_read,sys_read,sys32_read_wrapper)
SYSCALL(sys_write,sys_write,sys32_write_wrapper) SYSCALL(sys_write,sys_write,sys32_write_wrapper)
SYSCALL(sys_open,sys_open,sys32_open_wrapper) /* 5 */ SYSCALL(sys_open,sys_open,sys32_open_wrapper) /* 5 */
...@@ -19,7 +19,7 @@ SYSCALL(sys_restart_syscall,sys_restart_syscall,sys_restart_syscall) ...@@ -19,7 +19,7 @@ SYSCALL(sys_restart_syscall,sys_restart_syscall,sys_restart_syscall)
SYSCALL(sys_creat,sys_creat,sys32_creat_wrapper) SYSCALL(sys_creat,sys_creat,sys32_creat_wrapper)
SYSCALL(sys_link,sys_link,sys32_link_wrapper) SYSCALL(sys_link,sys_link,sys32_link_wrapper)
SYSCALL(sys_unlink,sys_unlink,sys32_unlink_wrapper) /* 10 */ SYSCALL(sys_unlink,sys_unlink,sys32_unlink_wrapper) /* 10 */
SYSCALL(sys_execve_glue,sys_execve_glue,sys32_execve_glue) SYSCALL(sys_execve,sys_execve,sys32_execve)
SYSCALL(sys_chdir,sys_chdir,sys32_chdir_wrapper) SYSCALL(sys_chdir,sys_chdir,sys32_chdir_wrapper)
SYSCALL(sys_time,sys_ni_syscall,sys32_time_wrapper) /* old time syscall */ SYSCALL(sys_time,sys_ni_syscall,sys32_time_wrapper) /* old time syscall */
SYSCALL(sys_mknod,sys_mknod,sys32_mknod_wrapper) SYSCALL(sys_mknod,sys_mknod,sys32_mknod_wrapper)
...@@ -127,8 +127,8 @@ SYSCALL(sys_swapoff,sys_swapoff,sys32_swapoff_wrapper) /* 115 */ ...@@ -127,8 +127,8 @@ SYSCALL(sys_swapoff,sys_swapoff,sys32_swapoff_wrapper) /* 115 */
SYSCALL(sys_sysinfo,sys_sysinfo,compat_sys_sysinfo_wrapper) SYSCALL(sys_sysinfo,sys_sysinfo,compat_sys_sysinfo_wrapper)
SYSCALL(sys_ipc,sys_ipc,sys32_ipc_wrapper) SYSCALL(sys_ipc,sys_ipc,sys32_ipc_wrapper)
SYSCALL(sys_fsync,sys_fsync,sys32_fsync_wrapper) SYSCALL(sys_fsync,sys_fsync,sys32_fsync_wrapper)
SYSCALL(sys_sigreturn_glue,sys_sigreturn_glue,sys32_sigreturn_glue) SYSCALL(sys_sigreturn,sys_sigreturn,sys32_sigreturn)
SYSCALL(sys_clone_glue,sys_clone_glue,sys32_clone_glue) /* 120 */ SYSCALL(sys_clone,sys_clone,sys32_clone) /* 120 */
SYSCALL(sys_setdomainname,sys_setdomainname,sys32_setdomainname_wrapper) SYSCALL(sys_setdomainname,sys_setdomainname,sys32_setdomainname_wrapper)
SYSCALL(sys_newuname,s390x_newuname,sys32_newuname_wrapper) SYSCALL(sys_newuname,s390x_newuname,sys32_newuname_wrapper)
NI_SYSCALL /* modify_ldt for i386 */ NI_SYSCALL /* modify_ldt for i386 */
...@@ -181,7 +181,7 @@ SYSCALL(sys_nfsservctl,sys_nfsservctl,compat_sys_nfsservctl_wrapper) ...@@ -181,7 +181,7 @@ SYSCALL(sys_nfsservctl,sys_nfsservctl,compat_sys_nfsservctl_wrapper)
SYSCALL(sys_setresgid16,sys_ni_syscall,sys32_setresgid16_wrapper) /* 170 old setresgid16 syscall */ SYSCALL(sys_setresgid16,sys_ni_syscall,sys32_setresgid16_wrapper) /* 170 old setresgid16 syscall */
SYSCALL(sys_getresgid16,sys_ni_syscall,sys32_getresgid16_wrapper) /* old getresgid16 syscall */ SYSCALL(sys_getresgid16,sys_ni_syscall,sys32_getresgid16_wrapper) /* old getresgid16 syscall */
SYSCALL(sys_prctl,sys_prctl,sys32_prctl_wrapper) SYSCALL(sys_prctl,sys_prctl,sys32_prctl_wrapper)
SYSCALL(sys_rt_sigreturn_glue,sys_rt_sigreturn_glue,sys32_rt_sigreturn_glue) SYSCALL(sys_rt_sigreturn,sys_rt_sigreturn,sys32_rt_sigreturn)
SYSCALL(sys_rt_sigaction,sys_rt_sigaction,sys32_rt_sigaction_wrapper) SYSCALL(sys_rt_sigaction,sys_rt_sigaction,sys32_rt_sigaction_wrapper)
SYSCALL(sys_rt_sigprocmask,sys_rt_sigprocmask,sys32_rt_sigprocmask_wrapper) /* 175 */ SYSCALL(sys_rt_sigprocmask,sys_rt_sigprocmask,sys32_rt_sigprocmask_wrapper) /* 175 */
SYSCALL(sys_rt_sigpending,sys_rt_sigpending,sys32_rt_sigpending_wrapper) SYSCALL(sys_rt_sigpending,sys_rt_sigpending,sys32_rt_sigpending_wrapper)
...@@ -194,11 +194,11 @@ SYSCALL(sys_chown16,sys_ni_syscall,sys32_chown16_wrapper) /* old chown16 syscall ...@@ -194,11 +194,11 @@ SYSCALL(sys_chown16,sys_ni_syscall,sys32_chown16_wrapper) /* old chown16 syscall
SYSCALL(sys_getcwd,sys_getcwd,sys32_getcwd_wrapper) SYSCALL(sys_getcwd,sys_getcwd,sys32_getcwd_wrapper)
SYSCALL(sys_capget,sys_capget,sys32_capget_wrapper) SYSCALL(sys_capget,sys_capget,sys32_capget_wrapper)
SYSCALL(sys_capset,sys_capset,sys32_capset_wrapper) /* 185 */ SYSCALL(sys_capset,sys_capset,sys32_capset_wrapper) /* 185 */
SYSCALL(sys_sigaltstack_glue,sys_sigaltstack_glue,sys32_sigaltstack_glue) SYSCALL(sys_sigaltstack,sys_sigaltstack,sys32_sigaltstack)
SYSCALL(sys_sendfile,sys_sendfile64,sys32_sendfile_wrapper) SYSCALL(sys_sendfile,sys_sendfile64,sys32_sendfile_wrapper)
NI_SYSCALL /* streams1 */ NI_SYSCALL /* streams1 */
NI_SYSCALL /* streams2 */ NI_SYSCALL /* streams2 */
SYSCALL(sys_vfork_glue,sys_vfork_glue,sys_vfork_glue) /* 190 */ SYSCALL(sys_vfork,sys_vfork,sys_vfork) /* 190 */
SYSCALL(sys_getrlimit,sys_getrlimit,compat_sys_getrlimit_wrapper) SYSCALL(sys_getrlimit,sys_getrlimit,compat_sys_getrlimit_wrapper)
SYSCALL(sys_mmap2,sys_mmap2,sys32_mmap2_wrapper) SYSCALL(sys_mmap2,sys_mmap2,sys32_mmap2_wrapper)
SYSCALL(sys_truncate64,sys_ni_syscall,sys32_truncate64_wrapper) SYSCALL(sys_truncate64,sys_ni_syscall,sys32_truncate64_wrapper)
......
...@@ -280,7 +280,6 @@ static void clock_comparator_interrupt(__u16 code) ...@@ -280,7 +280,6 @@ static void clock_comparator_interrupt(__u16 code)
} }
static void etr_reset(void); static void etr_reset(void);
static void etr_init(void);
static void etr_ext_handler(__u16); static void etr_ext_handler(__u16);
/* /*
...@@ -355,7 +354,6 @@ void __init time_init(void) ...@@ -355,7 +354,6 @@ void __init time_init(void)
#ifdef CONFIG_VIRT_TIMER #ifdef CONFIG_VIRT_TIMER
vtime_init(); vtime_init();
#endif #endif
etr_init();
} }
/* /*
...@@ -426,11 +424,11 @@ static struct etr_aib etr_port1; ...@@ -426,11 +424,11 @@ static struct etr_aib etr_port1;
static int etr_port1_uptodate; static int etr_port1_uptodate;
static unsigned long etr_events; static unsigned long etr_events;
static struct timer_list etr_timer; static struct timer_list etr_timer;
static struct tasklet_struct etr_tasklet;
static DEFINE_PER_CPU(atomic_t, etr_sync_word); static DEFINE_PER_CPU(atomic_t, etr_sync_word);
static void etr_timeout(unsigned long dummy); static void etr_timeout(unsigned long dummy);
static void etr_tasklet_fn(unsigned long dummy); static void etr_work_fn(struct work_struct *work);
static DECLARE_WORK(etr_work, etr_work_fn);
/* /*
* The etr get_clock function. It will write the current clock value * The etr get_clock function. It will write the current clock value
...@@ -507,29 +505,31 @@ static void etr_reset(void) ...@@ -507,29 +505,31 @@ static void etr_reset(void)
} }
} }
static void etr_init(void) static int __init etr_init(void)
{ {
struct etr_aib aib; struct etr_aib aib;
if (test_bit(ETR_FLAG_ENOSYS, &etr_flags)) if (test_bit(ETR_FLAG_ENOSYS, &etr_flags))
return; return 0;
/* Check if this machine has the steai instruction. */ /* Check if this machine has the steai instruction. */
if (etr_steai(&aib, ETR_STEAI_STEPPING_PORT) == 0) if (etr_steai(&aib, ETR_STEAI_STEPPING_PORT) == 0)
set_bit(ETR_FLAG_STEAI, &etr_flags); set_bit(ETR_FLAG_STEAI, &etr_flags);
setup_timer(&etr_timer, etr_timeout, 0UL); setup_timer(&etr_timer, etr_timeout, 0UL);
tasklet_init(&etr_tasklet, etr_tasklet_fn, 0);
if (!etr_port0_online && !etr_port1_online) if (!etr_port0_online && !etr_port1_online)
set_bit(ETR_FLAG_EACCES, &etr_flags); set_bit(ETR_FLAG_EACCES, &etr_flags);
if (etr_port0_online) { if (etr_port0_online) {
set_bit(ETR_EVENT_PORT0_CHANGE, &etr_events); set_bit(ETR_EVENT_PORT0_CHANGE, &etr_events);
tasklet_hi_schedule(&etr_tasklet); schedule_work(&etr_work);
} }
if (etr_port1_online) { if (etr_port1_online) {
set_bit(ETR_EVENT_PORT1_CHANGE, &etr_events); set_bit(ETR_EVENT_PORT1_CHANGE, &etr_events);
tasklet_hi_schedule(&etr_tasklet); schedule_work(&etr_work);
} }
return 0;
} }
arch_initcall(etr_init);
/* /*
* Two sorts of ETR machine checks. The architecture reads: * Two sorts of ETR machine checks. The architecture reads:
* "When a machine-check niterruption occurs and if a switch-to-local or * "When a machine-check niterruption occurs and if a switch-to-local or
...@@ -549,7 +549,7 @@ void etr_switch_to_local(void) ...@@ -549,7 +549,7 @@ void etr_switch_to_local(void)
return; return;
etr_disable_sync_clock(NULL); etr_disable_sync_clock(NULL);
set_bit(ETR_EVENT_SWITCH_LOCAL, &etr_events); set_bit(ETR_EVENT_SWITCH_LOCAL, &etr_events);
tasklet_hi_schedule(&etr_tasklet); schedule_work(&etr_work);
} }
/* /*
...@@ -564,7 +564,7 @@ void etr_sync_check(void) ...@@ -564,7 +564,7 @@ void etr_sync_check(void)
return; return;
etr_disable_sync_clock(NULL); etr_disable_sync_clock(NULL);
set_bit(ETR_EVENT_SYNC_CHECK, &etr_events); set_bit(ETR_EVENT_SYNC_CHECK, &etr_events);
tasklet_hi_schedule(&etr_tasklet); schedule_work(&etr_work);
} }
/* /*
...@@ -591,13 +591,13 @@ static void etr_ext_handler(__u16 code) ...@@ -591,13 +591,13 @@ static void etr_ext_handler(__u16 code)
* Both ports are not up-to-date now. * Both ports are not up-to-date now.
*/ */
set_bit(ETR_EVENT_PORT_ALERT, &etr_events); set_bit(ETR_EVENT_PORT_ALERT, &etr_events);
tasklet_hi_schedule(&etr_tasklet); schedule_work(&etr_work);
} }
static void etr_timeout(unsigned long dummy) static void etr_timeout(unsigned long dummy)
{ {
set_bit(ETR_EVENT_UPDATE, &etr_events); set_bit(ETR_EVENT_UPDATE, &etr_events);
tasklet_hi_schedule(&etr_tasklet); schedule_work(&etr_work);
} }
/* /*
...@@ -927,7 +927,7 @@ static struct etr_eacr etr_handle_update(struct etr_aib *aib, ...@@ -927,7 +927,7 @@ static struct etr_eacr etr_handle_update(struct etr_aib *aib,
if (!eacr.e0 && !eacr.e1) if (!eacr.e0 && !eacr.e1)
return eacr; return eacr;
/* Update port0 or port1 with aib stored in etr_tasklet_fn. */ /* Update port0 or port1 with aib stored in etr_work_fn. */
if (aib->esw.q == 0) { if (aib->esw.q == 0) {
/* Information for port 0 stored. */ /* Information for port 0 stored. */
if (eacr.p0 && !etr_port0_uptodate) { if (eacr.p0 && !etr_port0_uptodate) {
...@@ -1007,7 +1007,7 @@ static void etr_update_eacr(struct etr_eacr eacr) ...@@ -1007,7 +1007,7 @@ static void etr_update_eacr(struct etr_eacr eacr)
* particular this is the only function that calls etr_update_eacr(), * particular this is the only function that calls etr_update_eacr(),
* it "controls" the etr control register. * it "controls" the etr control register.
*/ */
static void etr_tasklet_fn(unsigned long dummy) static void etr_work_fn(struct work_struct *work)
{ {
unsigned long long now; unsigned long long now;
struct etr_eacr eacr; struct etr_eacr eacr;
...@@ -1220,13 +1220,13 @@ static ssize_t etr_online_store(struct sys_device *dev, ...@@ -1220,13 +1220,13 @@ static ssize_t etr_online_store(struct sys_device *dev,
return count; /* Nothing to do. */ return count; /* Nothing to do. */
etr_port0_online = value; etr_port0_online = value;
set_bit(ETR_EVENT_PORT0_CHANGE, &etr_events); set_bit(ETR_EVENT_PORT0_CHANGE, &etr_events);
tasklet_hi_schedule(&etr_tasklet); schedule_work(&etr_work);
} else { } else {
if (etr_port1_online == value) if (etr_port1_online == value)
return count; /* Nothing to do. */ return count; /* Nothing to do. */
etr_port1_online = value; etr_port1_online = value;
set_bit(ETR_EVENT_PORT1_CHANGE, &etr_events); set_bit(ETR_EVENT_PORT1_CHANGE, &etr_events);
tasklet_hi_schedule(&etr_tasklet); schedule_work(&etr_work);
} }
return count; return count;
} }
......
...@@ -30,7 +30,7 @@ ...@@ -30,7 +30,7 @@
#include <linux/kallsyms.h> #include <linux/kallsyms.h>
#include <linux/reboot.h> #include <linux/reboot.h>
#include <linux/kprobes.h> #include <linux/kprobes.h>
#include <linux/bug.h>
#include <asm/system.h> #include <asm/system.h>
#include <asm/uaccess.h> #include <asm/uaccess.h>
#include <asm/io.h> #include <asm/io.h>
...@@ -188,18 +188,31 @@ void dump_stack(void) ...@@ -188,18 +188,31 @@ void dump_stack(void)
EXPORT_SYMBOL(dump_stack); EXPORT_SYMBOL(dump_stack);
static inline int mask_bits(struct pt_regs *regs, unsigned long bits)
{
return (regs->psw.mask & bits) / ((~bits + 1) & bits);
}
void show_registers(struct pt_regs *regs) void show_registers(struct pt_regs *regs)
{ {
mm_segment_t old_fs;
char *mode; char *mode;
int i;
mode = (regs->psw.mask & PSW_MASK_PSTATE) ? "User" : "Krnl"; mode = (regs->psw.mask & PSW_MASK_PSTATE) ? "User" : "Krnl";
printk("%s PSW : %p %p", printk("%s PSW : %p %p",
mode, (void *) regs->psw.mask, mode, (void *) regs->psw.mask,
(void *) regs->psw.addr); (void *) regs->psw.addr);
print_symbol(" (%s)\n", regs->psw.addr & PSW_ADDR_INSN); print_symbol(" (%s)\n", regs->psw.addr & PSW_ADDR_INSN);
printk("%s GPRS: " FOURLONG, mode, printk(" R:%x T:%x IO:%x EX:%x Key:%x M:%x W:%x "
"P:%x AS:%x CC:%x PM:%x", mask_bits(regs, PSW_MASK_PER),
mask_bits(regs, PSW_MASK_DAT), mask_bits(regs, PSW_MASK_IO),
mask_bits(regs, PSW_MASK_EXT), mask_bits(regs, PSW_MASK_KEY),
mask_bits(regs, PSW_MASK_MCHECK), mask_bits(regs, PSW_MASK_WAIT),
mask_bits(regs, PSW_MASK_PSTATE), mask_bits(regs, PSW_MASK_ASC),
mask_bits(regs, PSW_MASK_CC), mask_bits(regs, PSW_MASK_PM));
#ifdef CONFIG_64BIT
printk(" EA:%x", mask_bits(regs, PSW_BASE_BITS));
#endif
printk("\n%s GPRS: " FOURLONG, mode,
regs->gprs[0], regs->gprs[1], regs->gprs[2], regs->gprs[3]); regs->gprs[0], regs->gprs[1], regs->gprs[2], regs->gprs[3]);
printk(" " FOURLONG, printk(" " FOURLONG,
regs->gprs[4], regs->gprs[5], regs->gprs[6], regs->gprs[7]); regs->gprs[4], regs->gprs[5], regs->gprs[6], regs->gprs[7]);
...@@ -208,41 +221,7 @@ void show_registers(struct pt_regs *regs) ...@@ -208,41 +221,7 @@ void show_registers(struct pt_regs *regs)
printk(" " FOURLONG, printk(" " FOURLONG,
regs->gprs[12], regs->gprs[13], regs->gprs[14], regs->gprs[15]); regs->gprs[12], regs->gprs[13], regs->gprs[14], regs->gprs[15]);
#if 0 show_code(regs);
/* FIXME: this isn't needed any more but it changes the ksymoops
* input. To remove or not to remove ... */
save_access_regs(regs->acrs);
printk("%s ACRS: %08x %08x %08x %08x\n", mode,
regs->acrs[0], regs->acrs[1], regs->acrs[2], regs->acrs[3]);
printk(" %08x %08x %08x %08x\n",
regs->acrs[4], regs->acrs[5], regs->acrs[6], regs->acrs[7]);
printk(" %08x %08x %08x %08x\n",
regs->acrs[8], regs->acrs[9], regs->acrs[10], regs->acrs[11]);
printk(" %08x %08x %08x %08x\n",
regs->acrs[12], regs->acrs[13], regs->acrs[14], regs->acrs[15]);
#endif
/*
* Print the first 20 byte of the instruction stream at the
* time of the fault.
*/
old_fs = get_fs();
if (regs->psw.mask & PSW_MASK_PSTATE)
set_fs(USER_DS);
else
set_fs(KERNEL_DS);
printk("%s Code: ", mode);
for (i = 0; i < 20; i++) {
unsigned char c;
if (__get_user(c, (char __user *)(regs->psw.addr + i))) {
printk(" Bad PSW.");
break;
}
printk("%02x ", c);
}
set_fs(old_fs);
printk("\n");
} }
/* This is called from fs/proc/array.c */ /* This is called from fs/proc/array.c */
...@@ -318,6 +297,11 @@ report_user_fault(long interruption_code, struct pt_regs *regs) ...@@ -318,6 +297,11 @@ report_user_fault(long interruption_code, struct pt_regs *regs)
#endif #endif
} }
int is_valid_bugaddr(unsigned long addr)
{
return 1;
}
static void __kprobes inline do_trap(long interruption_code, int signr, static void __kprobes inline do_trap(long interruption_code, int signr,
char *str, struct pt_regs *regs, char *str, struct pt_regs *regs,
siginfo_t *info) siginfo_t *info)
...@@ -344,9 +328,15 @@ static void __kprobes inline do_trap(long interruption_code, int signr, ...@@ -344,9 +328,15 @@ static void __kprobes inline do_trap(long interruption_code, int signr,
fixup = search_exception_tables(regs->psw.addr & PSW_ADDR_INSN); fixup = search_exception_tables(regs->psw.addr & PSW_ADDR_INSN);
if (fixup) if (fixup)
regs->psw.addr = fixup->fixup | PSW_ADDR_AMODE; regs->psw.addr = fixup->fixup | PSW_ADDR_AMODE;
else else {
enum bug_trap_type btt;
btt = report_bug(regs->psw.addr & PSW_ADDR_INSN);
if (btt == BUG_TRAP_TYPE_WARN)
return;
die(str, regs, interruption_code); die(str, regs, interruption_code);
} }
}
} }
static inline void __user *get_check_address(struct pt_regs *regs) static inline void __user *get_check_address(struct pt_regs *regs)
......
...@@ -45,6 +45,8 @@ SECTIONS ...@@ -45,6 +45,8 @@ SECTIONS
__ex_table : { *(__ex_table) } __ex_table : { *(__ex_table) }
__stop___ex_table = .; __stop___ex_table = .;
BUG_TABLE
.data : { /* Data */ .data : { /* Data */
*(.data) *(.data)
CONSTRUCTORS CONSTRUCTORS
...@@ -77,6 +79,12 @@ SECTIONS ...@@ -77,6 +79,12 @@ SECTIONS
*(.init.text) *(.init.text)
_einittext = .; _einittext = .;
} }
/*
* .exit.text is discarded at runtime, not link time,
* to deal with references from __bug_table
*/
.exit.text : { *(.exit.text) }
.init.data : { *(.init.data) } .init.data : { *(.init.data) }
. = ALIGN(256); . = ALIGN(256);
__setup_start = .; __setup_start = .;
...@@ -116,7 +124,7 @@ SECTIONS ...@@ -116,7 +124,7 @@ SECTIONS
/* Sections to be discarded */ /* Sections to be discarded */
/DISCARD/ : { /DISCARD/ : {
*(.exit.text) *(.exit.data) *(.exitcall.exit) *(.exit.data) *(.exitcall.exit)
} }
/* Stabs debugging sections. */ /* Stabs debugging sections. */
......
...@@ -128,7 +128,7 @@ static inline void set_vtimer(__u64 expires) ...@@ -128,7 +128,7 @@ static inline void set_vtimer(__u64 expires)
S390_lowcore.last_update_timer = expires; S390_lowcore.last_update_timer = expires;
/* store expire time for this CPU timer */ /* store expire time for this CPU timer */
per_cpu(virt_cpu_timer, smp_processor_id()).to_expire = expires; __get_cpu_var(virt_cpu_timer).to_expire = expires;
} }
#else #else
static inline void set_vtimer(__u64 expires) static inline void set_vtimer(__u64 expires)
...@@ -137,7 +137,7 @@ static inline void set_vtimer(__u64 expires) ...@@ -137,7 +137,7 @@ static inline void set_vtimer(__u64 expires)
asm volatile ("SPT %0" : : "m" (S390_lowcore.last_update_timer)); asm volatile ("SPT %0" : : "m" (S390_lowcore.last_update_timer));
/* store expire time for this CPU timer */ /* store expire time for this CPU timer */
per_cpu(virt_cpu_timer, smp_processor_id()).to_expire = expires; __get_cpu_var(virt_cpu_timer).to_expire = expires;
} }
#endif #endif
...@@ -145,7 +145,7 @@ static void start_cpu_timer(void) ...@@ -145,7 +145,7 @@ static void start_cpu_timer(void)
{ {
struct vtimer_queue *vt_list; struct vtimer_queue *vt_list;
vt_list = &per_cpu(virt_cpu_timer, smp_processor_id()); vt_list = &__get_cpu_var(virt_cpu_timer);
/* CPU timer interrupt is pending, don't reprogramm it */ /* CPU timer interrupt is pending, don't reprogramm it */
if (vt_list->idle & 1LL<<63) if (vt_list->idle & 1LL<<63)
...@@ -159,7 +159,7 @@ static void stop_cpu_timer(void) ...@@ -159,7 +159,7 @@ static void stop_cpu_timer(void)
{ {
struct vtimer_queue *vt_list; struct vtimer_queue *vt_list;
vt_list = &per_cpu(virt_cpu_timer, smp_processor_id()); vt_list = &__get_cpu_var(virt_cpu_timer);
/* nothing to do */ /* nothing to do */
if (list_empty(&vt_list->list)) { if (list_empty(&vt_list->list)) {
...@@ -219,7 +219,7 @@ static void do_callbacks(struct list_head *cb_list) ...@@ -219,7 +219,7 @@ static void do_callbacks(struct list_head *cb_list)
if (list_empty(cb_list)) if (list_empty(cb_list))
return; return;
vt_list = &per_cpu(virt_cpu_timer, smp_processor_id()); vt_list = &__get_cpu_var(virt_cpu_timer);
list_for_each_entry_safe(event, tmp, cb_list, entry) { list_for_each_entry_safe(event, tmp, cb_list, entry) {
fn = event->function; fn = event->function;
...@@ -244,7 +244,6 @@ static void do_callbacks(struct list_head *cb_list) ...@@ -244,7 +244,6 @@ static void do_callbacks(struct list_head *cb_list)
*/ */
static void do_cpu_timer_interrupt(__u16 error_code) static void do_cpu_timer_interrupt(__u16 error_code)
{ {
int cpu;
__u64 next, delta; __u64 next, delta;
struct vtimer_queue *vt_list; struct vtimer_queue *vt_list;
struct vtimer_list *event, *tmp; struct vtimer_list *event, *tmp;
...@@ -253,8 +252,7 @@ static void do_cpu_timer_interrupt(__u16 error_code) ...@@ -253,8 +252,7 @@ static void do_cpu_timer_interrupt(__u16 error_code)
struct list_head cb_list; struct list_head cb_list;
INIT_LIST_HEAD(&cb_list); INIT_LIST_HEAD(&cb_list);
cpu = smp_processor_id(); vt_list = &__get_cpu_var(virt_cpu_timer);
vt_list = &per_cpu(virt_cpu_timer, cpu);
/* walk timer list, fire all expired events */ /* walk timer list, fire all expired events */
spin_lock(&vt_list->lock); spin_lock(&vt_list->lock);
...@@ -534,7 +532,7 @@ void init_cpu_vtimer(void) ...@@ -534,7 +532,7 @@ void init_cpu_vtimer(void)
/* enable cpu timer interrupts */ /* enable cpu timer interrupts */
__ctl_set_bit(0,10); __ctl_set_bit(0,10);
vt_list = &per_cpu(virt_cpu_timer, smp_processor_id()); vt_list = &__get_cpu_var(virt_cpu_timer);
INIT_LIST_HEAD(&vt_list->list); INIT_LIST_HEAD(&vt_list->list);
spin_lock_init(&vt_list->lock); spin_lock_init(&vt_list->lock);
vt_list->to_expire = 0; vt_list->to_expire = 0;
......
This diff is collapsed.
...@@ -398,6 +398,9 @@ dasd_change_state(struct dasd_device *device) ...@@ -398,6 +398,9 @@ dasd_change_state(struct dasd_device *device)
if (device->state == device->target) if (device->state == device->target)
wake_up(&dasd_init_waitq); wake_up(&dasd_init_waitq);
/* let user-space know that the device status changed */
kobject_uevent(&device->cdev->dev.kobj, KOBJ_CHANGE);
} }
/* /*
......
...@@ -19,6 +19,7 @@ ...@@ -19,6 +19,7 @@
#include <asm/debug.h> #include <asm/debug.h>
#include <asm/uaccess.h> #include <asm/uaccess.h>
#include <asm/ipl.h>
/* This is ugly... */ /* This is ugly... */
#define PRINTK_HEADER "dasd_devmap:" #define PRINTK_HEADER "dasd_devmap:"
...@@ -133,6 +134,8 @@ dasd_call_setup(char *str) ...@@ -133,6 +134,8 @@ dasd_call_setup(char *str)
__setup ("dasd=", dasd_call_setup); __setup ("dasd=", dasd_call_setup);
#endif /* #ifndef MODULE */ #endif /* #ifndef MODULE */
#define DASD_IPLDEV "ipldev"
/* /*
* Read a device busid/devno from a string. * Read a device busid/devno from a string.
*/ */
...@@ -141,6 +144,20 @@ dasd_busid(char **str, int *id0, int *id1, int *devno) ...@@ -141,6 +144,20 @@ dasd_busid(char **str, int *id0, int *id1, int *devno)
{ {
int val, old_style; int val, old_style;
/* Interpret ipldev busid */
if (strncmp(DASD_IPLDEV, *str, strlen(DASD_IPLDEV)) == 0) {
if (ipl_info.type != IPL_TYPE_CCW) {
MESSAGE(KERN_ERR, "%s", "ipl device is not a ccw "
"device");
return -EINVAL;
}
*id0 = 0;
*id1 = ipl_info.data.ccw.dev_id.ssid;
*devno = ipl_info.data.ccw.dev_id.devno;
*str += strlen(DASD_IPLDEV);
return 0;
}
/* check for leading '0x' */ /* check for leading '0x' */
old_style = 0; old_style = 0;
if ((*str)[0] == '0' && (*str)[1] == 'x') { if ((*str)[0] == '0' && (*str)[1] == 'x') {
...@@ -828,6 +845,46 @@ dasd_discipline_show(struct device *dev, struct device_attribute *attr, ...@@ -828,6 +845,46 @@ dasd_discipline_show(struct device *dev, struct device_attribute *attr,
static DEVICE_ATTR(discipline, 0444, dasd_discipline_show, NULL); static DEVICE_ATTR(discipline, 0444, dasd_discipline_show, NULL);
static ssize_t
dasd_device_status_show(struct device *dev, struct device_attribute *attr,
char *buf)
{
struct dasd_device *device;
ssize_t len;
device = dasd_device_from_cdev(to_ccwdev(dev));
if (!IS_ERR(device)) {
switch (device->state) {
case DASD_STATE_NEW:
len = snprintf(buf, PAGE_SIZE, "new\n");
break;
case DASD_STATE_KNOWN:
len = snprintf(buf, PAGE_SIZE, "detected\n");
break;
case DASD_STATE_BASIC:
len = snprintf(buf, PAGE_SIZE, "basic\n");
break;
case DASD_STATE_UNFMT:
len = snprintf(buf, PAGE_SIZE, "unformatted\n");
break;
case DASD_STATE_READY:
len = snprintf(buf, PAGE_SIZE, "ready\n");
break;
case DASD_STATE_ONLINE:
len = snprintf(buf, PAGE_SIZE, "online\n");
break;
default:
len = snprintf(buf, PAGE_SIZE, "no stat\n");
break;
}
dasd_put_device(device);
} else
len = snprintf(buf, PAGE_SIZE, "unknown\n");
return len;
}
static DEVICE_ATTR(status, 0444, dasd_device_status_show, NULL);
static ssize_t static ssize_t
dasd_alias_show(struct device *dev, struct device_attribute *attr, char *buf) dasd_alias_show(struct device *dev, struct device_attribute *attr, char *buf)
{ {
...@@ -939,6 +996,7 @@ static DEVICE_ATTR(eer_enabled, 0644, dasd_eer_show, dasd_eer_store); ...@@ -939,6 +996,7 @@ static DEVICE_ATTR(eer_enabled, 0644, dasd_eer_show, dasd_eer_store);
static struct attribute * dasd_attrs[] = { static struct attribute * dasd_attrs[] = {
&dev_attr_readonly.attr, &dev_attr_readonly.attr,
&dev_attr_discipline.attr, &dev_attr_discipline.attr,
&dev_attr_status.attr,
&dev_attr_alias.attr, &dev_attr_alias.attr,
&dev_attr_vendor.attr, &dev_attr_vendor.attr,
&dev_attr_uid.attr, &dev_attr_uid.attr,
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
# #
obj-y += ctrlchar.o keyboard.o defkeymap.o sclp.o sclp_rw.o sclp_quiesce.o \ obj-y += ctrlchar.o keyboard.o defkeymap.o sclp.o sclp_rw.o sclp_quiesce.o \
sclp_info.o sclp_info.o sclp_config.o sclp_chp.o
obj-$(CONFIG_TN3270) += raw3270.o obj-$(CONFIG_TN3270) += raw3270.o
obj-$(CONFIG_TN3270_CONSOLE) += con3270.o obj-$(CONFIG_TN3270_CONSOLE) += con3270.o
...@@ -29,3 +29,6 @@ obj-$(CONFIG_S390_TAPE_34XX) += tape_34xx.o ...@@ -29,3 +29,6 @@ obj-$(CONFIG_S390_TAPE_34XX) += tape_34xx.o
obj-$(CONFIG_S390_TAPE_3590) += tape_3590.o obj-$(CONFIG_S390_TAPE_3590) += tape_3590.o
obj-$(CONFIG_MONREADER) += monreader.o obj-$(CONFIG_MONREADER) += monreader.o
obj-$(CONFIG_MONWRITER) += monwriter.o obj-$(CONFIG_MONWRITER) += monwriter.o
zcore_mod-objs := sclp_sdias.o zcore.o
obj-$(CONFIG_ZFCPDUMP) += zcore_mod.o
...@@ -813,12 +813,6 @@ con3215_unblank(void) ...@@ -813,12 +813,6 @@ con3215_unblank(void)
spin_unlock_irqrestore(get_ccwdev_lock(raw->cdev), flags); spin_unlock_irqrestore(get_ccwdev_lock(raw->cdev), flags);
} }
static int __init
con3215_consetup(struct console *co, char *options)
{
return 0;
}
/* /*
* The console structure for the 3215 console * The console structure for the 3215 console
*/ */
...@@ -827,7 +821,6 @@ static struct console con3215 = { ...@@ -827,7 +821,6 @@ static struct console con3215 = {
.write = con3215_write, .write = con3215_write,
.device = con3215_device, .device = con3215_device,
.unblank = con3215_unblank, .unblank = con3215_unblank,
.setup = con3215_consetup,
.flags = CON_PRINTBUFFER, .flags = CON_PRINTBUFFER,
}; };
......
...@@ -555,12 +555,6 @@ con3270_unblank(void) ...@@ -555,12 +555,6 @@ con3270_unblank(void)
spin_unlock_irqrestore(&cp->view.lock, flags); spin_unlock_irqrestore(&cp->view.lock, flags);
} }
static int __init
con3270_consetup(struct console *co, char *options)
{
return 0;
}
/* /*
* The console structure for the 3270 console * The console structure for the 3270 console
*/ */
...@@ -569,7 +563,6 @@ static struct console con3270 = { ...@@ -569,7 +563,6 @@ static struct console con3270 = {
.write = con3270_write, .write = con3270_write,
.device = con3270_device, .device = con3270_device,
.unblank = con3270_unblank, .unblank = con3270_unblank,
.setup = con3270_consetup,
.flags = CON_PRINTBUFFER, .flags = CON_PRINTBUFFER,
}; };
......
...@@ -15,6 +15,7 @@ ...@@ -15,6 +15,7 @@
#include <linux/timer.h> #include <linux/timer.h>
#include <linux/reboot.h> #include <linux/reboot.h>
#include <linux/jiffies.h> #include <linux/jiffies.h>
#include <linux/init.h>
#include <asm/types.h> #include <asm/types.h>
#include <asm/s390_ext.h> #include <asm/s390_ext.h>
...@@ -510,7 +511,7 @@ sclp_state_change_cb(struct evbuf_header *evbuf) ...@@ -510,7 +511,7 @@ sclp_state_change_cb(struct evbuf_header *evbuf)
} }
static struct sclp_register sclp_state_change_event = { static struct sclp_register sclp_state_change_event = {
.receive_mask = EvTyp_StateChange_Mask, .receive_mask = EVTYP_STATECHANGE_MASK,
.receiver_fn = sclp_state_change_cb .receiver_fn = sclp_state_change_cb
}; };
...@@ -930,3 +931,10 @@ sclp_init(void) ...@@ -930,3 +931,10 @@ sclp_init(void)
sclp_init_mask(1); sclp_init_mask(1);
return 0; return 0;
} }
static __init int sclp_initcall(void)
{
return sclp_init();
}
arch_initcall(sclp_initcall);
...@@ -19,33 +19,37 @@ ...@@ -19,33 +19,37 @@
#define MAX_KMEM_PAGES (sizeof(unsigned long) << 3) #define MAX_KMEM_PAGES (sizeof(unsigned long) << 3)
#define MAX_CONSOLE_PAGES 4 #define MAX_CONSOLE_PAGES 4
#define EvTyp_OpCmd 0x01 #define EVTYP_OPCMD 0x01
#define EvTyp_Msg 0x02 #define EVTYP_MSG 0x02
#define EvTyp_StateChange 0x08 #define EVTYP_STATECHANGE 0x08
#define EvTyp_PMsgCmd 0x09 #define EVTYP_PMSGCMD 0x09
#define EvTyp_CntlProgOpCmd 0x20 #define EVTYP_CNTLPROGOPCMD 0x20
#define EvTyp_CntlProgIdent 0x0B #define EVTYP_CNTLPROGIDENT 0x0B
#define EvTyp_SigQuiesce 0x1D #define EVTYP_SIGQUIESCE 0x1D
#define EvTyp_VT220Msg 0x1A #define EVTYP_VT220MSG 0x1A
#define EVTYP_CONFMGMDATA 0x04
#define EvTyp_OpCmd_Mask 0x80000000 #define EVTYP_SDIAS 0x1C
#define EvTyp_Msg_Mask 0x40000000
#define EvTyp_StateChange_Mask 0x01000000 #define EVTYP_OPCMD_MASK 0x80000000
#define EvTyp_PMsgCmd_Mask 0x00800000 #define EVTYP_MSG_MASK 0x40000000
#define EvTyp_CtlProgOpCmd_Mask 0x00000001 #define EVTYP_STATECHANGE_MASK 0x01000000
#define EvTyp_CtlProgIdent_Mask 0x00200000 #define EVTYP_PMSGCMD_MASK 0x00800000
#define EvTyp_SigQuiesce_Mask 0x00000008 #define EVTYP_CTLPROGOPCMD_MASK 0x00000001
#define EvTyp_VT220Msg_Mask 0x00000040 #define EVTYP_CTLPROGIDENT_MASK 0x00200000
#define EVTYP_SIGQUIESCE_MASK 0x00000008
#define GnrlMsgFlgs_DOM 0x8000 #define EVTYP_VT220MSG_MASK 0x00000040
#define GnrlMsgFlgs_SndAlrm 0x4000 #define EVTYP_CONFMGMDATA_MASK 0x10000000
#define GnrlMsgFlgs_HoldMsg 0x2000 #define EVTYP_SDIAS_MASK 0x00000010
#define LnTpFlgs_CntlText 0x8000 #define GNRLMSGFLGS_DOM 0x8000
#define LnTpFlgs_LabelText 0x4000 #define GNRLMSGFLGS_SNDALRM 0x4000
#define LnTpFlgs_DataText 0x2000 #define GNRLMSGFLGS_HOLDMSG 0x2000
#define LnTpFlgs_EndText 0x1000
#define LnTpFlgs_PromptText 0x0800 #define LNTPFLGS_CNTLTEXT 0x8000
#define LNTPFLGS_LABELTEXT 0x4000
#define LNTPFLGS_DATATEXT 0x2000
#define LNTPFLGS_ENDTEXT 0x1000
#define LNTPFLGS_PROMPTTEXT 0x0800
typedef unsigned int sclp_cmdw_t; typedef unsigned int sclp_cmdw_t;
...@@ -56,15 +60,15 @@ typedef unsigned int sclp_cmdw_t; ...@@ -56,15 +60,15 @@ typedef unsigned int sclp_cmdw_t;
#define SCLP_CMDW_READ_SCP_INFO_FORCED 0x00120001 #define SCLP_CMDW_READ_SCP_INFO_FORCED 0x00120001
#define GDS_ID_MDSMU 0x1310 #define GDS_ID_MDSMU 0x1310
#define GDS_ID_MDSRouteInfo 0x1311 #define GDS_ID_MDSROUTEINFO 0x1311
#define GDS_ID_AgUnWrkCorr 0x1549 #define GDS_ID_AGUNWRKCORR 0x1549
#define GDS_ID_SNACondReport 0x1532 #define GDS_ID_SNACONDREPORT 0x1532
#define GDS_ID_CPMSU 0x1212 #define GDS_ID_CPMSU 0x1212
#define GDS_ID_RoutTargInstr 0x154D #define GDS_ID_ROUTTARGINSTR 0x154D
#define GDS_ID_OpReq 0x8070 #define GDS_ID_OPREQ 0x8070
#define GDS_ID_TextCmd 0x1320 #define GDS_ID_TEXTCMD 0x1320
#define GDS_KEY_SelfDefTextMsg 0x31 #define GDS_KEY_SELFDEFTEXTMSG 0x31
typedef u32 sccb_mask_t; /* ATTENTION: assumes 32bit mask !!! */ typedef u32 sccb_mask_t; /* ATTENTION: assumes 32bit mask !!! */
......
/*
* drivers/s390/char/sclp_chp.c
*
* Copyright IBM Corp. 2007
* Author(s): Peter Oberparleiter <peter.oberparleiter@de.ibm.com>
*/
#include <linux/types.h>
#include <linux/gfp.h>
#include <linux/errno.h>
#include <linux/completion.h>
#include <asm/sclp.h>
#include <asm/chpid.h>
#include "sclp.h"
#define TAG "sclp_chp: "
#define SCLP_CMDW_CONFIGURE_CHANNEL_PATH 0x000f0001
#define SCLP_CMDW_DECONFIGURE_CHANNEL_PATH 0x000e0001
#define SCLP_CMDW_READ_CHANNEL_PATH_INFORMATION 0x00030001
static inline sclp_cmdw_t get_configure_cmdw(struct chp_id chpid)
{
return SCLP_CMDW_CONFIGURE_CHANNEL_PATH | chpid.id << 8;
}
static inline sclp_cmdw_t get_deconfigure_cmdw(struct chp_id chpid)
{
return SCLP_CMDW_DECONFIGURE_CHANNEL_PATH | chpid.id << 8;
}
static void chp_callback(struct sclp_req *req, void *data)
{
struct completion *completion = data;
complete(completion);
}
struct chp_cfg_sccb {
struct sccb_header header;
u8 ccm;
u8 reserved[6];
u8 cssid;
} __attribute__((packed));
struct chp_cfg_data {
struct chp_cfg_sccb sccb;
struct sclp_req req;
struct completion completion;
} __attribute__((packed));
static int do_configure(sclp_cmdw_t cmd)
{
struct chp_cfg_data *data;
int rc;
/* Prepare sccb. */
data = (struct chp_cfg_data *) get_zeroed_page(GFP_KERNEL | GFP_DMA);
if (!data)
return -ENOMEM;
data->sccb.header.length = sizeof(struct chp_cfg_sccb);
data->req.command = cmd;
data->req.sccb = &(data->sccb);
data->req.status = SCLP_REQ_FILLED;
data->req.callback = chp_callback;
data->req.callback_data = &(data->completion);
init_completion(&data->completion);
/* Perform sclp request. */
rc = sclp_add_request(&(data->req));
if (rc)
goto out;
wait_for_completion(&data->completion);
/* Check response .*/
if (data->req.status != SCLP_REQ_DONE) {
printk(KERN_WARNING TAG "configure channel-path request failed "
"(status=0x%02x)\n", data->req.status);
rc = -EIO;
goto out;
}
switch (data->sccb.header.response_code) {
case 0x0020:
case 0x0120:
case 0x0440:
case 0x0450:
break;
default:
printk(KERN_WARNING TAG "configure channel-path failed "
"(cmd=0x%08x, response=0x%04x)\n", cmd,
data->sccb.header.response_code);
rc = -EIO;
break;
}
out:
free_page((unsigned long) data);
return rc;
}
/**
* sclp_chp_configure - perform configure channel-path sclp command
* @chpid: channel-path ID
*
* Perform configure channel-path command sclp command for specified chpid.
* Return 0 after command successfully finished, non-zero otherwise.
*/
int sclp_chp_configure(struct chp_id chpid)
{
return do_configure(get_configure_cmdw(chpid));
}
/**
* sclp_chp_deconfigure - perform deconfigure channel-path sclp command
* @chpid: channel-path ID
*
* Perform deconfigure channel-path command sclp command for specified chpid
* and wait for completion. On success return 0. Return non-zero otherwise.
*/
int sclp_chp_deconfigure(struct chp_id chpid)
{
return do_configure(get_deconfigure_cmdw(chpid));
}
struct chp_info_sccb {
struct sccb_header header;
u8 recognized[SCLP_CHP_INFO_MASK_SIZE];
u8 standby[SCLP_CHP_INFO_MASK_SIZE];
u8 configured[SCLP_CHP_INFO_MASK_SIZE];
u8 ccm;
u8 reserved[6];
u8 cssid;
} __attribute__((packed));
struct chp_info_data {
struct chp_info_sccb sccb;
struct sclp_req req;
struct completion completion;
} __attribute__((packed));
/**
* sclp_chp_read_info - perform read channel-path information sclp command
* @info: resulting channel-path information data
*
* Perform read channel-path information sclp command and wait for completion.
* On success, store channel-path information in @info and return 0. Return
* non-zero otherwise.
*/
int sclp_chp_read_info(struct sclp_chp_info *info)
{
struct chp_info_data *data;
int rc;
/* Prepare sccb. */
data = (struct chp_info_data *) get_zeroed_page(GFP_KERNEL | GFP_DMA);
if (!data)
return -ENOMEM;
data->sccb.header.length = sizeof(struct chp_info_sccb);
data->req.command = SCLP_CMDW_READ_CHANNEL_PATH_INFORMATION;
data->req.sccb = &(data->sccb);
data->req.status = SCLP_REQ_FILLED;
data->req.callback = chp_callback;
data->req.callback_data = &(data->completion);
init_completion(&data->completion);
/* Perform sclp request. */
rc = sclp_add_request(&(data->req));
if (rc)
goto out;
wait_for_completion(&data->completion);
/* Check response .*/
if (data->req.status != SCLP_REQ_DONE) {
printk(KERN_WARNING TAG "read channel-path info request failed "
"(status=0x%02x)\n", data->req.status);
rc = -EIO;
goto out;
}
if (data->sccb.header.response_code != 0x0010) {
printk(KERN_WARNING TAG "read channel-path info failed "
"(response=0x%04x)\n", data->sccb.header.response_code);
rc = -EIO;
goto out;
}
memcpy(info->recognized, data->sccb.recognized,
SCLP_CHP_INFO_MASK_SIZE);
memcpy(info->standby, data->sccb.standby,
SCLP_CHP_INFO_MASK_SIZE);
memcpy(info->configured, data->sccb.configured,
SCLP_CHP_INFO_MASK_SIZE);
out:
free_page((unsigned long) data);
return rc;
}
/*
* drivers/s390/char/sclp_config.c
*
* Copyright IBM Corp. 2007
* Author(s): Heiko Carstens <heiko.carstens@de.ibm.com>
*/
#include <linux/init.h>
#include <linux/errno.h>
#include <linux/cpu.h>
#include <linux/sysdev.h>
#include <linux/workqueue.h>
#include "sclp.h"
#define TAG "sclp_config: "
struct conf_mgm_data {
u8 reserved;
u8 ev_qualifier;
} __attribute__((packed));
#define EV_QUAL_CAP_CHANGE 3
static struct work_struct sclp_cpu_capability_work;
static void sclp_cpu_capability_notify(struct work_struct *work)
{
int cpu;
struct sys_device *sysdev;
printk(KERN_WARNING TAG "cpu capability changed.\n");
lock_cpu_hotplug();
for_each_online_cpu(cpu) {
sysdev = get_cpu_sysdev(cpu);
kobject_uevent(&sysdev->kobj, KOBJ_CHANGE);
}
unlock_cpu_hotplug();
}
static void sclp_conf_receiver_fn(struct evbuf_header *evbuf)
{
struct conf_mgm_data *cdata;
cdata = (struct conf_mgm_data *)(evbuf + 1);
if (cdata->ev_qualifier == EV_QUAL_CAP_CHANGE)
schedule_work(&sclp_cpu_capability_work);
}
static struct sclp_register sclp_conf_register =
{
.receive_mask = EVTYP_CONFMGMDATA_MASK,
.receiver_fn = sclp_conf_receiver_fn,
};
static int __init sclp_conf_init(void)
{
int rc;
INIT_WORK(&sclp_cpu_capability_work, sclp_cpu_capability_notify);
rc = sclp_register(&sclp_conf_register);
if (rc) {
printk(KERN_ERR TAG "failed to register (%d).\n", rc);
return rc;
}
if (!(sclp_conf_register.sclp_receive_mask & EVTYP_CONFMGMDATA_MASK)) {
printk(KERN_WARNING TAG "no configuration management.\n");
sclp_unregister(&sclp_conf_register);
rc = -ENOSYS;
}
return rc;
}
__initcall(sclp_conf_init);
...@@ -46,7 +46,7 @@ struct cpi_sccb { ...@@ -46,7 +46,7 @@ struct cpi_sccb {
/* Event type structure for write message and write priority message */ /* Event type structure for write message and write priority message */
static struct sclp_register sclp_cpi_event = static struct sclp_register sclp_cpi_event =
{ {
.send_mask = EvTyp_CtlProgIdent_Mask .send_mask = EVTYP_CTLPROGIDENT_MASK
}; };
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
...@@ -201,7 +201,7 @@ cpi_module_init(void) ...@@ -201,7 +201,7 @@ cpi_module_init(void)
"console.\n"); "console.\n");
return -EINVAL; return -EINVAL;
} }
if (!(sclp_cpi_event.sclp_send_mask & EvTyp_CtlProgIdent_Mask)) { if (!(sclp_cpi_event.sclp_send_mask & EVTYP_CTLPROGIDENT_MASK)) {
printk(KERN_WARNING "cpi: no control program identification " printk(KERN_WARNING "cpi: no control program identification "
"support\n"); "support\n");
sclp_unregister(&sclp_cpi_event); sclp_unregister(&sclp_cpi_event);
......
...@@ -43,7 +43,7 @@ sclp_quiesce_handler(struct evbuf_header *evbuf) ...@@ -43,7 +43,7 @@ sclp_quiesce_handler(struct evbuf_header *evbuf)
} }
static struct sclp_register sclp_quiesce_event = { static struct sclp_register sclp_quiesce_event = {
.receive_mask = EvTyp_SigQuiesce_Mask, .receive_mask = EVTYP_SIGQUIESCE_MASK,
.receiver_fn = sclp_quiesce_handler .receiver_fn = sclp_quiesce_handler
}; };
......
...@@ -30,7 +30,7 @@ ...@@ -30,7 +30,7 @@
/* Event type structure for write message and write priority message */ /* Event type structure for write message and write priority message */
static struct sclp_register sclp_rw_event = { static struct sclp_register sclp_rw_event = {
.send_mask = EvTyp_Msg_Mask | EvTyp_PMsgCmd_Mask .send_mask = EVTYP_MSG_MASK | EVTYP_PMSGCMD_MASK
}; };
/* /*
...@@ -64,7 +64,7 @@ sclp_make_buffer(void *page, unsigned short columns, unsigned short htab) ...@@ -64,7 +64,7 @@ sclp_make_buffer(void *page, unsigned short columns, unsigned short htab)
memset(sccb, 0, sizeof(struct write_sccb)); memset(sccb, 0, sizeof(struct write_sccb));
sccb->header.length = sizeof(struct write_sccb); sccb->header.length = sizeof(struct write_sccb);
sccb->msg_buf.header.length = sizeof(struct msg_buf); sccb->msg_buf.header.length = sizeof(struct msg_buf);
sccb->msg_buf.header.type = EvTyp_Msg; sccb->msg_buf.header.type = EVTYP_MSG;
sccb->msg_buf.mdb.header.length = sizeof(struct mdb); sccb->msg_buf.mdb.header.length = sizeof(struct mdb);
sccb->msg_buf.mdb.header.type = 1; sccb->msg_buf.mdb.header.type = 1;
sccb->msg_buf.mdb.header.tag = 0xD4C4C240; /* ebcdic "MDB " */ sccb->msg_buf.mdb.header.tag = 0xD4C4C240; /* ebcdic "MDB " */
...@@ -114,7 +114,7 @@ sclp_initialize_mto(struct sclp_buffer *buffer, int max_len) ...@@ -114,7 +114,7 @@ sclp_initialize_mto(struct sclp_buffer *buffer, int max_len)
memset(mto, 0, sizeof(struct mto)); memset(mto, 0, sizeof(struct mto));
mto->length = sizeof(struct mto); mto->length = sizeof(struct mto);
mto->type = 4; /* message text object */ mto->type = 4; /* message text object */
mto->line_type_flags = LnTpFlgs_EndText; /* end text */ mto->line_type_flags = LNTPFLGS_ENDTEXT; /* end text */
/* set pointer to first byte after struct mto. */ /* set pointer to first byte after struct mto. */
buffer->current_line = (char *) (mto + 1); buffer->current_line = (char *) (mto + 1);
...@@ -215,7 +215,7 @@ sclp_write(struct sclp_buffer *buffer, const unsigned char *msg, int count) ...@@ -215,7 +215,7 @@ sclp_write(struct sclp_buffer *buffer, const unsigned char *msg, int count)
case '\a': /* bell, one for several times */ case '\a': /* bell, one for several times */
/* set SCLP sound alarm bit in General Object */ /* set SCLP sound alarm bit in General Object */
buffer->sccb->msg_buf.mdb.go.general_msg_flags |= buffer->sccb->msg_buf.mdb.go.general_msg_flags |=
GnrlMsgFlgs_SndAlrm; GNRLMSGFLGS_SNDALRM;
break; break;
case '\t': /* horizontal tabulator */ case '\t': /* horizontal tabulator */
/* check if new mto needs to be created */ /* check if new mto needs to be created */
...@@ -452,12 +452,12 @@ sclp_emit_buffer(struct sclp_buffer *buffer, ...@@ -452,12 +452,12 @@ sclp_emit_buffer(struct sclp_buffer *buffer,
return -EIO; return -EIO;
sccb = buffer->sccb; sccb = buffer->sccb;
if (sclp_rw_event.sclp_send_mask & EvTyp_Msg_Mask) if (sclp_rw_event.sclp_send_mask & EVTYP_MSG_MASK)
/* Use normal write message */ /* Use normal write message */
sccb->msg_buf.header.type = EvTyp_Msg; sccb->msg_buf.header.type = EVTYP_MSG;
else if (sclp_rw_event.sclp_send_mask & EvTyp_PMsgCmd_Mask) else if (sclp_rw_event.sclp_send_mask & EVTYP_PMSGCMD_MASK)
/* Use write priority message */ /* Use write priority message */
sccb->msg_buf.header.type = EvTyp_PMsgCmd; sccb->msg_buf.header.type = EVTYP_PMSGCMD;
else else
return -ENOSYS; return -ENOSYS;
buffer->request.command = SCLP_CMDW_WRITE_EVENT_DATA; buffer->request.command = SCLP_CMDW_WRITE_EVENT_DATA;
......
This diff is collapsed.
...@@ -648,7 +648,7 @@ sclp_eval_textcmd(struct gds_subvector *start, ...@@ -648,7 +648,7 @@ sclp_eval_textcmd(struct gds_subvector *start,
subvec = start; subvec = start;
while (subvec < end) { while (subvec < end) {
subvec = find_gds_subvector(subvec, end, subvec = find_gds_subvector(subvec, end,
GDS_KEY_SelfDefTextMsg); GDS_KEY_SELFDEFTEXTMSG);
if (!subvec) if (!subvec)
break; break;
sclp_eval_selfdeftextmsg((struct gds_subvector *)(subvec + 1), sclp_eval_selfdeftextmsg((struct gds_subvector *)(subvec + 1),
...@@ -664,7 +664,7 @@ sclp_eval_cpmsu(struct gds_vector *start, struct gds_vector *end) ...@@ -664,7 +664,7 @@ sclp_eval_cpmsu(struct gds_vector *start, struct gds_vector *end)
vec = start; vec = start;
while (vec < end) { while (vec < end) {
vec = find_gds_vector(vec, end, GDS_ID_TextCmd); vec = find_gds_vector(vec, end, GDS_ID_TEXTCMD);
if (!vec) if (!vec)
break; break;
sclp_eval_textcmd((struct gds_subvector *)(vec + 1), sclp_eval_textcmd((struct gds_subvector *)(vec + 1),
...@@ -703,7 +703,7 @@ sclp_tty_state_change(struct sclp_register *reg) ...@@ -703,7 +703,7 @@ sclp_tty_state_change(struct sclp_register *reg)
static struct sclp_register sclp_input_event = static struct sclp_register sclp_input_event =
{ {
.receive_mask = EvTyp_OpCmd_Mask | EvTyp_PMsgCmd_Mask, .receive_mask = EVTYP_OPCMD_MASK | EVTYP_PMSGCMD_MASK,
.state_change_fn = sclp_tty_state_change, .state_change_fn = sclp_tty_state_change,
.receiver_fn = sclp_tty_receiver .receiver_fn = sclp_tty_receiver
}; };
......
...@@ -99,8 +99,8 @@ static void sclp_vt220_emit_current(void); ...@@ -99,8 +99,8 @@ static void sclp_vt220_emit_current(void);
/* Registration structure for our interest in SCLP event buffers */ /* Registration structure for our interest in SCLP event buffers */
static struct sclp_register sclp_vt220_register = { static struct sclp_register sclp_vt220_register = {
.send_mask = EvTyp_VT220Msg_Mask, .send_mask = EVTYP_VT220MSG_MASK,
.receive_mask = EvTyp_VT220Msg_Mask, .receive_mask = EVTYP_VT220MSG_MASK,
.state_change_fn = NULL, .state_change_fn = NULL,
.receiver_fn = sclp_vt220_receiver_fn .receiver_fn = sclp_vt220_receiver_fn
}; };
...@@ -202,7 +202,7 @@ sclp_vt220_callback(struct sclp_req *request, void *data) ...@@ -202,7 +202,7 @@ sclp_vt220_callback(struct sclp_req *request, void *data)
static int static int
__sclp_vt220_emit(struct sclp_vt220_request *request) __sclp_vt220_emit(struct sclp_vt220_request *request)
{ {
if (!(sclp_vt220_register.sclp_send_mask & EvTyp_VT220Msg_Mask)) { if (!(sclp_vt220_register.sclp_send_mask & EVTYP_VT220MSG_MASK)) {
request->sclp_req.status = SCLP_REQ_FAILED; request->sclp_req.status = SCLP_REQ_FAILED;
return -EIO; return -EIO;
} }
...@@ -284,7 +284,7 @@ sclp_vt220_initialize_page(void *page) ...@@ -284,7 +284,7 @@ sclp_vt220_initialize_page(void *page)
sccb->header.length = sizeof(struct sclp_vt220_sccb); sccb->header.length = sizeof(struct sclp_vt220_sccb);
sccb->header.function_code = SCLP_NORMAL_WRITE; sccb->header.function_code = SCLP_NORMAL_WRITE;
sccb->header.response_code = 0x0000; sccb->header.response_code = 0x0000;
sccb->evbuf.type = EvTyp_VT220Msg; sccb->evbuf.type = EVTYP_VT220MSG;
sccb->evbuf.length = sizeof(struct evbuf_header); sccb->evbuf.length = sizeof(struct evbuf_header);
return request; return request;
......
...@@ -125,7 +125,7 @@ static struct vmlogrdr_priv_t sys_ser[] = { ...@@ -125,7 +125,7 @@ static struct vmlogrdr_priv_t sys_ser[] = {
.recording_name = "EREP", .recording_name = "EREP",
.minor_num = 0, .minor_num = 0,
.buffer_free = 1, .buffer_free = 1,
.priv_lock = SPIN_LOCK_UNLOCKED, .priv_lock = __SPIN_LOCK_UNLOCKED(sys_ser[0].priv_lock),
.autorecording = 1, .autorecording = 1,
.autopurge = 1, .autopurge = 1,
}, },
...@@ -134,7 +134,7 @@ static struct vmlogrdr_priv_t sys_ser[] = { ...@@ -134,7 +134,7 @@ static struct vmlogrdr_priv_t sys_ser[] = {
.recording_name = "ACCOUNT", .recording_name = "ACCOUNT",
.minor_num = 1, .minor_num = 1,
.buffer_free = 1, .buffer_free = 1,
.priv_lock = SPIN_LOCK_UNLOCKED, .priv_lock = __SPIN_LOCK_UNLOCKED(sys_ser[1].priv_lock),
.autorecording = 1, .autorecording = 1,
.autopurge = 1, .autopurge = 1,
}, },
...@@ -143,7 +143,7 @@ static struct vmlogrdr_priv_t sys_ser[] = { ...@@ -143,7 +143,7 @@ static struct vmlogrdr_priv_t sys_ser[] = {
.recording_name = "SYMPTOM", .recording_name = "SYMPTOM",
.minor_num = 2, .minor_num = 2,
.buffer_free = 1, .buffer_free = 1,
.priv_lock = SPIN_LOCK_UNLOCKED, .priv_lock = __SPIN_LOCK_UNLOCKED(sys_ser[2].priv_lock),
.autorecording = 1, .autorecording = 1,
.autopurge = 1, .autopurge = 1,
} }
...@@ -385,6 +385,9 @@ static int vmlogrdr_release (struct inode *inode, struct file *filp) ...@@ -385,6 +385,9 @@ static int vmlogrdr_release (struct inode *inode, struct file *filp)
struct vmlogrdr_priv_t * logptr = filp->private_data; struct vmlogrdr_priv_t * logptr = filp->private_data;
iucv_path_sever(logptr->path, NULL);
kfree(logptr->path);
logptr->path = NULL;
if (logptr->autorecording) { if (logptr->autorecording) {
ret = vmlogrdr_recording(logptr,0,logptr->autopurge); ret = vmlogrdr_recording(logptr,0,logptr->autopurge);
if (ret) if (ret)
......
This diff is collapsed.
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
# Makefile for the S/390 common i/o drivers # Makefile for the S/390 common i/o drivers
# #
obj-y += airq.o blacklist.o chsc.o cio.o css.o obj-y += airq.o blacklist.o chsc.o cio.o css.o chp.o idset.o
ccw_device-objs += device.o device_fsm.o device_ops.o ccw_device-objs += device.o device_fsm.o device_ops.o
ccw_device-objs += device_id.o device_pgid.o device_status.o ccw_device-objs += device_id.o device_pgid.o device_status.o
obj-y += ccw_device.o cmf.o obj-y += ccw_device.o cmf.o
......
...@@ -75,8 +75,10 @@ static void ccwgroup_ungroup_callback(struct device *dev) ...@@ -75,8 +75,10 @@ static void ccwgroup_ungroup_callback(struct device *dev)
{ {
struct ccwgroup_device *gdev = to_ccwgroupdev(dev); struct ccwgroup_device *gdev = to_ccwgroupdev(dev);
mutex_lock(&gdev->reg_mutex);
__ccwgroup_remove_symlinks(gdev); __ccwgroup_remove_symlinks(gdev);
device_unregister(dev); device_unregister(dev);
mutex_unlock(&gdev->reg_mutex);
} }
static ssize_t static ssize_t
...@@ -173,7 +175,8 @@ ccwgroup_create(struct device *root, ...@@ -173,7 +175,8 @@ ccwgroup_create(struct device *root,
return -ENOMEM; return -ENOMEM;
atomic_set(&gdev->onoff, 0); atomic_set(&gdev->onoff, 0);
mutex_init(&gdev->reg_mutex);
mutex_lock(&gdev->reg_mutex);
for (i = 0; i < argc; i++) { for (i = 0; i < argc; i++) {
gdev->cdev[i] = get_ccwdev_by_busid(cdrv, argv[i]); gdev->cdev[i] = get_ccwdev_by_busid(cdrv, argv[i]);
...@@ -183,12 +186,12 @@ ccwgroup_create(struct device *root, ...@@ -183,12 +186,12 @@ ccwgroup_create(struct device *root,
|| gdev->cdev[i]->id.driver_info != || gdev->cdev[i]->id.driver_info !=
gdev->cdev[0]->id.driver_info) { gdev->cdev[0]->id.driver_info) {
rc = -EINVAL; rc = -EINVAL;
goto free_dev; goto error;
} }
/* Don't allow a device to belong to more than one group. */ /* Don't allow a device to belong to more than one group. */
if (gdev->cdev[i]->dev.driver_data) { if (gdev->cdev[i]->dev.driver_data) {
rc = -EINVAL; rc = -EINVAL;
goto free_dev; goto error;
} }
gdev->cdev[i]->dev.driver_data = gdev; gdev->cdev[i]->dev.driver_data = gdev;
} }
...@@ -203,9 +206,8 @@ ccwgroup_create(struct device *root, ...@@ -203,9 +206,8 @@ ccwgroup_create(struct device *root,
gdev->cdev[0]->dev.bus_id); gdev->cdev[0]->dev.bus_id);
rc = device_register(&gdev->dev); rc = device_register(&gdev->dev);
if (rc) if (rc)
goto free_dev; goto error;
get_device(&gdev->dev); get_device(&gdev->dev);
rc = device_create_file(&gdev->dev, &dev_attr_ungroup); rc = device_create_file(&gdev->dev, &dev_attr_ungroup);
...@@ -216,27 +218,21 @@ ccwgroup_create(struct device *root, ...@@ -216,27 +218,21 @@ ccwgroup_create(struct device *root,
rc = __ccwgroup_create_symlinks(gdev); rc = __ccwgroup_create_symlinks(gdev);
if (!rc) { if (!rc) {
mutex_unlock(&gdev->reg_mutex);
put_device(&gdev->dev); put_device(&gdev->dev);
return 0; return 0;
} }
device_remove_file(&gdev->dev, &dev_attr_ungroup); device_remove_file(&gdev->dev, &dev_attr_ungroup);
device_unregister(&gdev->dev); device_unregister(&gdev->dev);
error: error:
for (i = 0; i < argc; i++)
if (gdev->cdev[i]) {
put_device(&gdev->cdev[i]->dev);
gdev->cdev[i]->dev.driver_data = NULL;
}
put_device(&gdev->dev);
return rc;
free_dev:
for (i = 0; i < argc; i++) for (i = 0; i < argc; i++)
if (gdev->cdev[i]) { if (gdev->cdev[i]) {
if (gdev->cdev[i]->dev.driver_data == gdev) if (gdev->cdev[i]->dev.driver_data == gdev)
gdev->cdev[i]->dev.driver_data = NULL; gdev->cdev[i]->dev.driver_data = NULL;
put_device(&gdev->cdev[i]->dev); put_device(&gdev->cdev[i]->dev);
} }
kfree(gdev); mutex_unlock(&gdev->reg_mutex);
put_device(&gdev->dev);
return rc; return rc;
} }
...@@ -422,8 +418,12 @@ ccwgroup_driver_unregister (struct ccwgroup_driver *cdriver) ...@@ -422,8 +418,12 @@ ccwgroup_driver_unregister (struct ccwgroup_driver *cdriver)
get_driver(&cdriver->driver); get_driver(&cdriver->driver);
while ((dev = driver_find_device(&cdriver->driver, NULL, NULL, while ((dev = driver_find_device(&cdriver->driver, NULL, NULL,
__ccwgroup_match_all))) { __ccwgroup_match_all))) {
__ccwgroup_remove_symlinks(to_ccwgroupdev(dev)); struct ccwgroup_device *gdev = to_ccwgroupdev(dev);
mutex_lock(&gdev->reg_mutex);
__ccwgroup_remove_symlinks(gdev);
device_unregister(dev); device_unregister(dev);
mutex_unlock(&gdev->reg_mutex);
put_device(dev); put_device(dev);
} }
put_driver(&cdriver->driver); put_driver(&cdriver->driver);
...@@ -444,8 +444,10 @@ __ccwgroup_get_gdev_by_cdev(struct ccw_device *cdev) ...@@ -444,8 +444,10 @@ __ccwgroup_get_gdev_by_cdev(struct ccw_device *cdev)
if (cdev->dev.driver_data) { if (cdev->dev.driver_data) {
gdev = (struct ccwgroup_device *)cdev->dev.driver_data; gdev = (struct ccwgroup_device *)cdev->dev.driver_data;
if (get_device(&gdev->dev)) { if (get_device(&gdev->dev)) {
mutex_lock(&gdev->reg_mutex);
if (device_is_registered(&gdev->dev)) if (device_is_registered(&gdev->dev))
return gdev; return gdev;
mutex_unlock(&gdev->reg_mutex);
put_device(&gdev->dev); put_device(&gdev->dev);
} }
return NULL; return NULL;
...@@ -465,6 +467,7 @@ ccwgroup_remove_ccwdev(struct ccw_device *cdev) ...@@ -465,6 +467,7 @@ ccwgroup_remove_ccwdev(struct ccw_device *cdev)
if (gdev) { if (gdev) {
__ccwgroup_remove_symlinks(gdev); __ccwgroup_remove_symlinks(gdev);
device_unregister(&gdev->dev); device_unregister(&gdev->dev);
mutex_unlock(&gdev->reg_mutex);
put_device(&gdev->dev); put_device(&gdev->dev);
} }
} }
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
...@@ -476,7 +476,7 @@ struct cmb_area { ...@@ -476,7 +476,7 @@ struct cmb_area {
}; };
static struct cmb_area cmb_area = { static struct cmb_area cmb_area = {
.lock = SPIN_LOCK_UNLOCKED, .lock = __SPIN_LOCK_UNLOCKED(cmb_area.lock),
.list = LIST_HEAD_INIT(cmb_area.list), .list = LIST_HEAD_INIT(cmb_area.list),
.num_channels = 1024, .num_channels = 1024,
}; };
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
...@@ -11,6 +11,7 @@ struct ccwgroup_device { ...@@ -11,6 +11,7 @@ struct ccwgroup_device {
CCWGROUP_ONLINE, CCWGROUP_ONLINE,
} state; } state;
atomic_t onoff; atomic_t onoff;
struct mutex reg_mutex;
unsigned int count; /* number of attached slave devices */ unsigned int count; /* number of attached slave devices */
struct device dev; /* master device */ struct device dev; /* master device */
struct ccw_device *cdev[0]; /* variable number, allocate as needed */ struct ccw_device *cdev[0]; /* variable number, allocate as needed */
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment