提交 0029b4d1 编写于 作者: A Alexander Egorenkov 提交者: Heiko Carstens

s390/sclp: use only one sclp early buffer to send commands

A buffer that can be used for communication with SCLP is required
to lie below 2GB memory address. Therefore, both sclp_info_sccb
and sclp_early_sccb must fulfill this requirement if passed directly
to the sclp_early_cmd() function. Instead, use only sclp_early_sccb
for communication with SCLP. This allows the buffer sclp_info_sccb
to be placed anywhere in the memory address space and, therefore,
simplifies the process of making the decompressor relocatable later on,
one thing less to relocate. And make sure that the length of the new unified
early SCLP buffer is no less than the length of the removed sclp_info_sccb
buffer which might be larger than the length of the sclp_early_sccb buffer.
Signed-off-by: NAlexander Egorenkov <egorenar@linux.ibm.com>
Acked-by: NHeiko Carstens <hca@linux.ibm.com>
Reviewed-by: NVasily Gorbik <gor@linux.ibm.com>
Signed-off-by: NHeiko Carstens <hca@linux.ibm.com>
上级 6040b3f4
...@@ -27,6 +27,7 @@ ...@@ -27,6 +27,7 @@
#include <asm/asm-offsets.h> #include <asm/asm-offsets.h>
#include <asm/page.h> #include <asm/page.h>
#include <asm/ptrace.h> #include <asm/ptrace.h>
#include <asm/sclp.h>
#define ARCH_OFFSET 4 #define ARCH_OFFSET 4
...@@ -410,6 +411,6 @@ SYM_DATA_START(parmarea) ...@@ -410,6 +411,6 @@ SYM_DATA_START(parmarea)
SYM_DATA_END(parmarea) SYM_DATA_END(parmarea)
.org EARLY_SCCB_OFFSET .org EARLY_SCCB_OFFSET
.fill 4096 .fill EXT_SCCB_READ_SCP
.org HEAD_END .org HEAD_END
...@@ -8,8 +8,6 @@ ...@@ -8,8 +8,6 @@
#define _ASM_S390_SCLP_H #define _ASM_S390_SCLP_H
#include <linux/types.h> #include <linux/types.h>
#include <asm/chpid.h>
#include <asm/cpu.h>
#define SCLP_CHP_INFO_MASK_SIZE 32 #define SCLP_CHP_INFO_MASK_SIZE 32
#define EARLY_SCCB_SIZE PAGE_SIZE #define EARLY_SCCB_SIZE PAGE_SIZE
...@@ -19,6 +17,10 @@ ...@@ -19,6 +17,10 @@
/* 24 + 16 * SCLP_MAX_CORES */ /* 24 + 16 * SCLP_MAX_CORES */
#define EXT_SCCB_READ_CPU (3 * PAGE_SIZE) #define EXT_SCCB_READ_CPU (3 * PAGE_SIZE)
#ifndef __ASSEMBLY__
#include <asm/chpid.h>
#include <asm/cpu.h>
struct sclp_chp_info { struct sclp_chp_info {
u8 recognized[SCLP_CHP_INFO_MASK_SIZE]; u8 recognized[SCLP_CHP_INFO_MASK_SIZE];
u8 standby[SCLP_CHP_INFO_MASK_SIZE]; u8 standby[SCLP_CHP_INFO_MASK_SIZE];
...@@ -147,4 +149,5 @@ static inline int sclp_get_core_info(struct sclp_core_info *info, int early) ...@@ -147,4 +149,5 @@ static inline int sclp_get_core_info(struct sclp_core_info *info, int early)
return _sclp_get_core_info(info); return _sclp_get_core_info(info);
} }
#endif /* __ASSEMBLY__ */
#endif /* _ASM_S390_SCLP_H */ #endif /* _ASM_S390_SCLP_H */
...@@ -7,6 +7,7 @@ ...@@ -7,6 +7,7 @@
#define _ASM_S390_SETUP_H #define _ASM_S390_SETUP_H
#include <linux/bits.h> #include <linux/bits.h>
#include <asm/sclp.h>
#include <uapi/asm/setup.h> #include <uapi/asm/setup.h>
#include <linux/build_bug.h> #include <linux/build_bug.h>
...@@ -14,7 +15,7 @@ ...@@ -14,7 +15,7 @@
#define EP_STRING "S390EP" #define EP_STRING "S390EP"
#define PARMAREA 0x10400 #define PARMAREA 0x10400
#define EARLY_SCCB_OFFSET 0x11000 #define EARLY_SCCB_OFFSET 0x11000
#define HEAD_END 0x12000 #define HEAD_END (EARLY_SCCB_OFFSET + EXT_SCCB_READ_SCP)
/* /*
* Machine features detected in early.c * Machine features detected in early.c
......
...@@ -235,11 +235,20 @@ void sclp_early_printk(const char *str) ...@@ -235,11 +235,20 @@ void sclp_early_printk(const char *str)
__sclp_early_printk(str, strlen(str)); __sclp_early_printk(str, strlen(str));
} }
/*
* We can't pass sclp_info_sccb to sclp_early_cmd() here directly,
* because it might not fulfil the requiremets for a SCLP communication buffer:
* - lie below 2G in memory
* - be page-aligned
* Therefore, we use the buffer sclp_early_sccb (which fulfils all those
* requirements) temporarily for communication and copy a received response
* back into the buffer sclp_info_sccb upon successful completion.
*/
int __init sclp_early_read_info(void) int __init sclp_early_read_info(void)
{ {
int i; int i;
int length = test_facility(140) ? EXT_SCCB_READ_SCP : PAGE_SIZE; int length = test_facility(140) ? EXT_SCCB_READ_SCP : PAGE_SIZE;
struct read_info_sccb *sccb = &sclp_info_sccb; struct read_info_sccb *sccb = (struct read_info_sccb *)sclp_early_sccb;
sclp_cmdw_t commands[] = {SCLP_CMDW_READ_SCP_INFO_FORCED, sclp_cmdw_t commands[] = {SCLP_CMDW_READ_SCP_INFO_FORCED,
SCLP_CMDW_READ_SCP_INFO}; SCLP_CMDW_READ_SCP_INFO};
...@@ -251,6 +260,7 @@ int __init sclp_early_read_info(void) ...@@ -251,6 +260,7 @@ int __init sclp_early_read_info(void)
if (sclp_early_cmd(commands[i], sccb)) if (sclp_early_cmd(commands[i], sccb))
break; break;
if (sccb->header.response_code == 0x10) { if (sccb->header.response_code == 0x10) {
memcpy(&sclp_info_sccb, sccb, length);
sclp_info_sccb_valid = 1; sclp_info_sccb_valid = 1;
return 0; return 0;
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册