Commit 684d2fd4 authored by Hendrik Brueckner's avatar Hendrik Brueckner Committed by Martin Schwidefsky

[S390] kernel: Append scpdata to kernel boot command line

Append scpdata to the kernel boot command line. If scpdata starts
with the equal sign (=), the kernel boot command line is replaced.
(For consistency with zIPL and IPL PARM parameters.)

To use scpdata for the kernel boot command line, scpdata must consist
of ascii characters only. If scpdata contains other characters,
scpdata is not appended to the kernel boot command line.
In addition, re-IPL is extended for setting scpdata for the next
Linux reboot.
Signed-off-by: default avatarHendrik Brueckner <brueckner@linux.vnet.ibm.com>
Signed-off-by: default avatarMartin Schwidefsky <schwidefsky@de.ibm.com>
parent 6292b9ef
...@@ -57,6 +57,8 @@ struct ipl_block_fcp { ...@@ -57,6 +57,8 @@ struct ipl_block_fcp {
} __attribute__((packed)); } __attribute__((packed));
#define DIAG308_VMPARM_SIZE 64 #define DIAG308_VMPARM_SIZE 64
#define DIAG308_SCPDATA_SIZE (PAGE_SIZE - (sizeof(struct ipl_list_hdr) + \
offsetof(struct ipl_block_fcp, scp_data)))
struct ipl_block_ccw { struct ipl_block_ccw {
u8 load_parm[8]; u8 load_parm[8];
...@@ -91,7 +93,8 @@ extern void do_halt(void); ...@@ -91,7 +93,8 @@ extern void do_halt(void);
extern void do_poff(void); extern void do_poff(void);
extern void ipl_save_parameters(void); extern void ipl_save_parameters(void);
extern void ipl_update_parameters(void); extern void ipl_update_parameters(void);
extern void get_ipl_vmparm(char *); extern size_t append_ipl_vmparm(char *, size_t);
extern size_t append_ipl_scpdata(char *, size_t);
enum { enum {
IPL_DEVNO_VALID = 1, IPL_DEVNO_VALID = 1,
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
#ifndef _ASM_S390_SETUP_H #ifndef _ASM_S390_SETUP_H
#define _ASM_S390_SETUP_H #define _ASM_S390_SETUP_H
#define COMMAND_LINE_SIZE 1024 #define COMMAND_LINE_SIZE 4096
#define ARCH_COMMAND_LINE_SIZE 896 #define ARCH_COMMAND_LINE_SIZE 896
......
...@@ -81,6 +81,8 @@ asm( ...@@ -81,6 +81,8 @@ asm(
" br 14\n" " br 14\n"
" .size savesys_ipl_nss, .-savesys_ipl_nss\n"); " .size savesys_ipl_nss, .-savesys_ipl_nss\n");
static __initdata char upper_command_line[COMMAND_LINE_SIZE];
static noinline __init void create_kernel_nss(void) static noinline __init void create_kernel_nss(void)
{ {
unsigned int i, stext_pfn, eshared_pfn, end_pfn, min_size; unsigned int i, stext_pfn, eshared_pfn, end_pfn, min_size;
...@@ -90,7 +92,6 @@ static noinline __init void create_kernel_nss(void) ...@@ -90,7 +92,6 @@ static noinline __init void create_kernel_nss(void)
int response; int response;
size_t len; size_t len;
char *savesys_ptr; char *savesys_ptr;
char upper_command_line[COMMAND_LINE_SIZE];
char defsys_cmd[DEFSYS_CMD_SIZE]; char defsys_cmd[DEFSYS_CMD_SIZE];
char savesys_cmd[SAVESYS_CMD_SIZE]; char savesys_cmd[SAVESYS_CMD_SIZE];
...@@ -367,21 +368,35 @@ static __init void rescue_initrd(void) ...@@ -367,21 +368,35 @@ static __init void rescue_initrd(void)
} }
/* Set up boot command line */ /* Set up boot command line */
static void __init setup_boot_command_line(void) static void __init append_to_cmdline(size_t (*ipl_data)(char *, size_t))
{ {
char *parm = NULL; char *parm, *delim;
size_t rc, len;
len = strlen(boot_command_line);
delim = boot_command_line + len; /* '\0' character position */
parm = boot_command_line + len + 1; /* append right after '\0' */
rc = ipl_data(parm, COMMAND_LINE_SIZE - len - 1);
if (rc) {
if (*parm == '=')
memmove(boot_command_line, parm + 1, rc);
else
*delim = ' '; /* replace '\0' with space */
}
}
static void __init setup_boot_command_line(void)
{
/* copy arch command line */ /* copy arch command line */
strlcpy(boot_command_line, COMMAND_LINE, ARCH_COMMAND_LINE_SIZE); strlcpy(boot_command_line, COMMAND_LINE, ARCH_COMMAND_LINE_SIZE);
/* append IPL PARM data to the boot command line */ /* append IPL PARM data to the boot command line */
if (MACHINE_IS_VM) { if (MACHINE_IS_VM)
parm = boot_command_line + strlen(boot_command_line); append_to_cmdline(append_ipl_vmparm);
*parm++ = ' ';
get_ipl_vmparm(parm); append_to_cmdline(append_ipl_scpdata);
if (parm[0] == '=')
memmove(boot_command_line, parm + 1, strlen(parm));
}
} }
......
...@@ -272,17 +272,18 @@ static ssize_t ipl_type_show(struct kobject *kobj, struct kobj_attribute *attr, ...@@ -272,17 +272,18 @@ static ssize_t ipl_type_show(struct kobject *kobj, struct kobj_attribute *attr,
static struct kobj_attribute sys_ipl_type_attr = __ATTR_RO(ipl_type); static struct kobj_attribute sys_ipl_type_attr = __ATTR_RO(ipl_type);
/* VM IPL PARM routines */ /* VM IPL PARM routines */
static void reipl_get_ascii_vmparm(char *dest, size_t reipl_get_ascii_vmparm(char *dest, size_t size,
const struct ipl_parameter_block *ipb) const struct ipl_parameter_block *ipb)
{ {
int i; int i;
int len = 0; size_t len;
char has_lowercase = 0; char has_lowercase = 0;
len = 0;
if ((ipb->ipl_info.ccw.vm_flags & DIAG308_VM_FLAGS_VP_VALID) && if ((ipb->ipl_info.ccw.vm_flags & DIAG308_VM_FLAGS_VP_VALID) &&
(ipb->ipl_info.ccw.vm_parm_len > 0)) { (ipb->ipl_info.ccw.vm_parm_len > 0)) {
len = ipb->ipl_info.ccw.vm_parm_len; len = min_t(size_t, size - 1, ipb->ipl_info.ccw.vm_parm_len);
memcpy(dest, ipb->ipl_info.ccw.vm_parm, len); memcpy(dest, ipb->ipl_info.ccw.vm_parm, len);
/* If at least one character is lowercase, we assume mixed /* If at least one character is lowercase, we assume mixed
* case; otherwise we convert everything to lowercase. * case; otherwise we convert everything to lowercase.
...@@ -299,14 +300,20 @@ static void reipl_get_ascii_vmparm(char *dest, ...@@ -299,14 +300,20 @@ static void reipl_get_ascii_vmparm(char *dest,
EBCASC(dest, len); EBCASC(dest, len);
} }
dest[len] = 0; dest[len] = 0;
return len;
} }
void get_ipl_vmparm(char *dest) size_t append_ipl_vmparm(char *dest, size_t size)
{ {
size_t rc;
rc = 0;
if (diag308_set_works && (ipl_block.hdr.pbt == DIAG308_IPL_TYPE_CCW)) if (diag308_set_works && (ipl_block.hdr.pbt == DIAG308_IPL_TYPE_CCW))
reipl_get_ascii_vmparm(dest, &ipl_block); rc = reipl_get_ascii_vmparm(dest, size, &ipl_block);
else else
dest[0] = 0; dest[0] = 0;
return rc;
} }
static ssize_t ipl_vm_parm_show(struct kobject *kobj, static ssize_t ipl_vm_parm_show(struct kobject *kobj,
...@@ -314,10 +321,56 @@ static ssize_t ipl_vm_parm_show(struct kobject *kobj, ...@@ -314,10 +321,56 @@ static ssize_t ipl_vm_parm_show(struct kobject *kobj,
{ {
char parm[DIAG308_VMPARM_SIZE + 1] = {}; char parm[DIAG308_VMPARM_SIZE + 1] = {};
get_ipl_vmparm(parm); append_ipl_vmparm(parm, sizeof(parm));
return sprintf(page, "%s\n", parm); return sprintf(page, "%s\n", parm);
} }
static size_t scpdata_length(const char* buf, size_t count)
{
while (count) {
if (buf[count - 1] != '\0' && buf[count - 1] != ' ')
break;
count--;
}
return count;
}
size_t reipl_append_ascii_scpdata(char *dest, size_t size,
const struct ipl_parameter_block *ipb)
{
size_t count;
size_t i;
count = min(size - 1, scpdata_length(ipb->ipl_info.fcp.scp_data,
ipb->ipl_info.fcp.scp_data_len));
if (!count)
goto out;
for (i = 0; i < count; i++)
if (!isascii(ipb->ipl_info.fcp.scp_data[i])) {
count = 0;
goto out;
}
memcpy(dest, ipb->ipl_info.fcp.scp_data, count);
out:
dest[count] = '\0';
return count;
}
size_t append_ipl_scpdata(char *dest, size_t len)
{
size_t rc;
rc = 0;
if (ipl_block.hdr.pbt == DIAG308_IPL_TYPE_FCP)
rc = reipl_append_ascii_scpdata(dest, len, &ipl_block);
else
dest[0] = 0;
return rc;
}
static struct kobj_attribute sys_ipl_vm_parm_attr = static struct kobj_attribute sys_ipl_vm_parm_attr =
__ATTR(parm, S_IRUGO, ipl_vm_parm_show, NULL); __ATTR(parm, S_IRUGO, ipl_vm_parm_show, NULL);
...@@ -553,7 +606,7 @@ static ssize_t reipl_generic_vmparm_show(struct ipl_parameter_block *ipb, ...@@ -553,7 +606,7 @@ static ssize_t reipl_generic_vmparm_show(struct ipl_parameter_block *ipb,
{ {
char vmparm[DIAG308_VMPARM_SIZE + 1] = {}; char vmparm[DIAG308_VMPARM_SIZE + 1] = {};
reipl_get_ascii_vmparm(vmparm, ipb); reipl_get_ascii_vmparm(vmparm, sizeof(vmparm), ipb);
return sprintf(page, "%s\n", vmparm); return sprintf(page, "%s\n", vmparm);
} }
...@@ -626,6 +679,59 @@ static struct kobj_attribute sys_reipl_ccw_vmparm_attr = ...@@ -626,6 +679,59 @@ static struct kobj_attribute sys_reipl_ccw_vmparm_attr =
/* FCP reipl device attributes */ /* FCP reipl device attributes */
static ssize_t reipl_fcp_scpdata_read(struct kobject *kobj,
struct bin_attribute *attr,
char *buf, loff_t off, size_t count)
{
size_t size = reipl_block_fcp->ipl_info.fcp.scp_data_len;
void *scp_data = reipl_block_fcp->ipl_info.fcp.scp_data;
return memory_read_from_buffer(buf, count, &off, scp_data, size);
}
static ssize_t reipl_fcp_scpdata_write(struct kobject *kobj,
struct bin_attribute *attr,
char *buf, loff_t off, size_t count)
{
size_t padding;
size_t scpdata_len;
if (off < 0)
return -EINVAL;
if (off >= DIAG308_SCPDATA_SIZE)
return -ENOSPC;
if (count > DIAG308_SCPDATA_SIZE - off)
count = DIAG308_SCPDATA_SIZE - off;
memcpy(reipl_block_fcp->ipl_info.fcp.scp_data, buf + off, count);
scpdata_len = off + count;
if (scpdata_len % 8) {
padding = 8 - (scpdata_len % 8);
memset(reipl_block_fcp->ipl_info.fcp.scp_data + scpdata_len,
0, padding);
scpdata_len += padding;
}
reipl_block_fcp->ipl_info.fcp.scp_data_len = scpdata_len;
reipl_block_fcp->hdr.len = IPL_PARM_BLK_FCP_LEN + scpdata_len;
reipl_block_fcp->hdr.blk0_len = IPL_PARM_BLK0_FCP_LEN + scpdata_len;
return count;
}
static struct bin_attribute sys_reipl_fcp_scp_data_attr = {
.attr = {
.name = "scp_data",
.mode = S_IRUGO | S_IWUSR,
},
.size = PAGE_SIZE,
.read = reipl_fcp_scpdata_read,
.write = reipl_fcp_scpdata_write,
};
DEFINE_IPL_ATTR_RW(reipl_fcp, wwpn, "0x%016llx\n", "%016llx\n", DEFINE_IPL_ATTR_RW(reipl_fcp, wwpn, "0x%016llx\n", "%016llx\n",
reipl_block_fcp->ipl_info.fcp.wwpn); reipl_block_fcp->ipl_info.fcp.wwpn);
DEFINE_IPL_ATTR_RW(reipl_fcp, lun, "0x%016llx\n", "%016llx\n", DEFINE_IPL_ATTR_RW(reipl_fcp, lun, "0x%016llx\n", "%016llx\n",
...@@ -647,7 +753,6 @@ static struct attribute *reipl_fcp_attrs[] = { ...@@ -647,7 +753,6 @@ static struct attribute *reipl_fcp_attrs[] = {
}; };
static struct attribute_group reipl_fcp_attr_group = { static struct attribute_group reipl_fcp_attr_group = {
.name = IPL_FCP_STR,
.attrs = reipl_fcp_attrs, .attrs = reipl_fcp_attrs,
}; };
...@@ -895,6 +1000,7 @@ static struct kobj_attribute reipl_type_attr = ...@@ -895,6 +1000,7 @@ static struct kobj_attribute reipl_type_attr =
__ATTR(reipl_type, 0644, reipl_type_show, reipl_type_store); __ATTR(reipl_type, 0644, reipl_type_show, reipl_type_store);
static struct kset *reipl_kset; static struct kset *reipl_kset;
static struct kset *reipl_fcp_kset;
static void get_ipl_string(char *dst, struct ipl_parameter_block *ipb, static void get_ipl_string(char *dst, struct ipl_parameter_block *ipb,
const enum ipl_method m) const enum ipl_method m)
...@@ -906,7 +1012,7 @@ static void get_ipl_string(char *dst, struct ipl_parameter_block *ipb, ...@@ -906,7 +1012,7 @@ static void get_ipl_string(char *dst, struct ipl_parameter_block *ipb,
reipl_get_ascii_loadparm(loadparm, ipb); reipl_get_ascii_loadparm(loadparm, ipb);
reipl_get_ascii_nss_name(nss_name, ipb); reipl_get_ascii_nss_name(nss_name, ipb);
reipl_get_ascii_vmparm(vmparm, ipb); reipl_get_ascii_vmparm(vmparm, sizeof(vmparm), ipb);
switch (m) { switch (m) {
case REIPL_METHOD_CCW_VM: case REIPL_METHOD_CCW_VM:
...@@ -1076,23 +1182,44 @@ static int __init reipl_fcp_init(void) ...@@ -1076,23 +1182,44 @@ static int __init reipl_fcp_init(void)
int rc; int rc;
if (!diag308_set_works) { if (!diag308_set_works) {
if (ipl_info.type == IPL_TYPE_FCP) if (ipl_info.type == IPL_TYPE_FCP) {
make_attrs_ro(reipl_fcp_attrs); make_attrs_ro(reipl_fcp_attrs);
else sys_reipl_fcp_scp_data_attr.attr.mode = S_IRUGO;
} else
return 0; return 0;
} }
reipl_block_fcp = (void *) get_zeroed_page(GFP_KERNEL); reipl_block_fcp = (void *) get_zeroed_page(GFP_KERNEL);
if (!reipl_block_fcp) if (!reipl_block_fcp)
return -ENOMEM; return -ENOMEM;
rc = sysfs_create_group(&reipl_kset->kobj, &reipl_fcp_attr_group);
/* sysfs: create fcp kset for mixing attr group and bin attrs */
reipl_fcp_kset = kset_create_and_add(IPL_FCP_STR, NULL,
&reipl_kset->kobj);
if (!reipl_kset) {
free_page((unsigned long) reipl_block_fcp);
return -ENOMEM;
}
rc = sysfs_create_group(&reipl_fcp_kset->kobj, &reipl_fcp_attr_group);
if (rc) { if (rc) {
free_page((unsigned long)reipl_block_fcp); kset_unregister(reipl_fcp_kset);
free_page((unsigned long) reipl_block_fcp);
return rc; return rc;
} }
if (ipl_info.type == IPL_TYPE_FCP) {
rc = sysfs_create_bin_file(&reipl_fcp_kset->kobj,
&sys_reipl_fcp_scp_data_attr);
if (rc) {
sysfs_remove_group(&reipl_fcp_kset->kobj, &reipl_fcp_attr_group);
kset_unregister(reipl_fcp_kset);
free_page((unsigned long) reipl_block_fcp);
return rc;
}
if (ipl_info.type == IPL_TYPE_FCP)
memcpy(reipl_block_fcp, IPL_PARMBLOCK_START, PAGE_SIZE); memcpy(reipl_block_fcp, IPL_PARMBLOCK_START, PAGE_SIZE);
} else { else {
reipl_block_fcp->hdr.len = IPL_PARM_BLK_FCP_LEN; reipl_block_fcp->hdr.len = IPL_PARM_BLK_FCP_LEN;
reipl_block_fcp->hdr.version = IPL_PARM_BLOCK_VERSION; reipl_block_fcp->hdr.version = IPL_PARM_BLOCK_VERSION;
reipl_block_fcp->hdr.blk0_len = IPL_PARM_BLK0_FCP_LEN; reipl_block_fcp->hdr.blk0_len = IPL_PARM_BLK0_FCP_LEN;
......
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