diff options
| -rw-r--r-- | arch/s390/kernel/early.c | 9 | ||||
| -rw-r--r-- | arch/s390/kernel/entry.S | 8 | ||||
| -rw-r--r-- | arch/s390/kernel/entry64.S | 8 | ||||
| -rw-r--r-- | drivers/s390/char/monreader.c | 1 | ||||
| -rw-r--r-- | drivers/s390/char/sclp_quiesce.c | 48 |
5 files changed, 56 insertions, 18 deletions
diff --git a/arch/s390/kernel/early.c b/arch/s390/kernel/early.c index bf8b4ae7ff2d..e49e9e0c69fd 100644 --- a/arch/s390/kernel/early.c +++ b/arch/s390/kernel/early.c | |||
| @@ -55,6 +55,7 @@ static void __init reset_tod_clock(void) | |||
| 55 | disabled_wait(0); | 55 | disabled_wait(0); |
| 56 | 56 | ||
| 57 | sched_clock_base_cc = TOD_UNIX_EPOCH; | 57 | sched_clock_base_cc = TOD_UNIX_EPOCH; |
| 58 | S390_lowcore.last_update_clock = sched_clock_base_cc; | ||
| 58 | } | 59 | } |
| 59 | 60 | ||
| 60 | #ifdef CONFIG_SHARED_KERNEL | 61 | #ifdef CONFIG_SHARED_KERNEL |
| @@ -167,6 +168,14 @@ static noinline __init void create_kernel_nss(void) | |||
| 167 | return; | 168 | return; |
| 168 | } | 169 | } |
| 169 | 170 | ||
| 171 | /* re-initialize cputime accounting. */ | ||
| 172 | sched_clock_base_cc = get_clock(); | ||
| 173 | S390_lowcore.last_update_clock = sched_clock_base_cc; | ||
| 174 | S390_lowcore.last_update_timer = 0x7fffffffffffffffULL; | ||
| 175 | S390_lowcore.user_timer = 0; | ||
| 176 | S390_lowcore.system_timer = 0; | ||
| 177 | asm volatile("SPT 0(%0)" : : "a" (&S390_lowcore.last_update_timer)); | ||
| 178 | |||
| 170 | /* re-setup boot command line with new ipl vm parms */ | 179 | /* re-setup boot command line with new ipl vm parms */ |
| 171 | ipl_update_parameters(); | 180 | ipl_update_parameters(); |
| 172 | setup_boot_command_line(); | 181 | setup_boot_command_line(); |
diff --git a/arch/s390/kernel/entry.S b/arch/s390/kernel/entry.S index f43d2ee54464..48215d15762b 100644 --- a/arch/s390/kernel/entry.S +++ b/arch/s390/kernel/entry.S | |||
| @@ -565,10 +565,10 @@ pgm_svcper: | |||
| 565 | lh %r7,0x8a # get svc number from lowcore | 565 | lh %r7,0x8a # get svc number from lowcore |
| 566 | l %r9,__LC_THREAD_INFO # load pointer to thread_info struct | 566 | l %r9,__LC_THREAD_INFO # load pointer to thread_info struct |
| 567 | TRACE_IRQS_OFF | 567 | TRACE_IRQS_OFF |
| 568 | l %r1,__TI_task(%r9) | 568 | l %r8,__TI_task(%r9) |
| 569 | mvc __THREAD_per+__PER_atmid(2,%r1),__LC_PER_ATMID | 569 | mvc __THREAD_per+__PER_atmid(2,%r8),__LC_PER_ATMID |
| 570 | mvc __THREAD_per+__PER_address(4,%r1),__LC_PER_ADDRESS | 570 | mvc __THREAD_per+__PER_address(4,%r8),__LC_PER_ADDRESS |
| 571 | mvc __THREAD_per+__PER_access_id(1,%r1),__LC_PER_ACCESS_ID | 571 | mvc __THREAD_per+__PER_access_id(1,%r8),__LC_PER_ACCESS_ID |
| 572 | oi __TI_flags+3(%r9),_TIF_SINGLE_STEP # set TIF_SINGLE_STEP | 572 | oi __TI_flags+3(%r9),_TIF_SINGLE_STEP # set TIF_SINGLE_STEP |
| 573 | TRACE_IRQS_ON | 573 | TRACE_IRQS_ON |
| 574 | stosm __SF_EMPTY(%r15),0x03 # reenable interrupts | 574 | stosm __SF_EMPTY(%r15),0x03 # reenable interrupts |
diff --git a/arch/s390/kernel/entry64.S b/arch/s390/kernel/entry64.S index a6f7b20df616..9aff1d449b6e 100644 --- a/arch/s390/kernel/entry64.S +++ b/arch/s390/kernel/entry64.S | |||
| @@ -543,10 +543,10 @@ pgm_svcper: | |||
| 543 | mvc __LC_LAST_UPDATE_TIMER(8),__LC_SYNC_ENTER_TIMER | 543 | mvc __LC_LAST_UPDATE_TIMER(8),__LC_SYNC_ENTER_TIMER |
| 544 | llgh %r7,__LC_SVC_INT_CODE # get svc number from lowcore | 544 | llgh %r7,__LC_SVC_INT_CODE # get svc number from lowcore |
| 545 | lg %r9,__LC_THREAD_INFO # load pointer to thread_info struct | 545 | lg %r9,__LC_THREAD_INFO # load pointer to thread_info struct |
| 546 | lg %r1,__TI_task(%r9) | 546 | lg %r8,__TI_task(%r9) |
| 547 | mvc __THREAD_per+__PER_atmid(2,%r1),__LC_PER_ATMID | 547 | mvc __THREAD_per+__PER_atmid(2,%r8),__LC_PER_ATMID |
| 548 | mvc __THREAD_per+__PER_address(8,%r1),__LC_PER_ADDRESS | 548 | mvc __THREAD_per+__PER_address(8,%r8),__LC_PER_ADDRESS |
| 549 | mvc __THREAD_per+__PER_access_id(1,%r1),__LC_PER_ACCESS_ID | 549 | mvc __THREAD_per+__PER_access_id(1,%r8),__LC_PER_ACCESS_ID |
| 550 | oi __TI_flags+7(%r9),_TIF_SINGLE_STEP # set TIF_SINGLE_STEP | 550 | oi __TI_flags+7(%r9),_TIF_SINGLE_STEP # set TIF_SINGLE_STEP |
| 551 | TRACE_IRQS_ON | 551 | TRACE_IRQS_ON |
| 552 | stosm __SF_EMPTY(%r15),0x03 # reenable interrupts | 552 | stosm __SF_EMPTY(%r15),0x03 # reenable interrupts |
diff --git a/drivers/s390/char/monreader.c b/drivers/s390/char/monreader.c index 89ece1c235aa..66e21dd23154 100644 --- a/drivers/s390/char/monreader.c +++ b/drivers/s390/char/monreader.c | |||
| @@ -357,6 +357,7 @@ static int mon_close(struct inode *inode, struct file *filp) | |||
| 357 | atomic_set(&monpriv->msglim_count, 0); | 357 | atomic_set(&monpriv->msglim_count, 0); |
| 358 | monpriv->write_index = 0; | 358 | monpriv->write_index = 0; |
| 359 | monpriv->read_index = 0; | 359 | monpriv->read_index = 0; |
| 360 | dev_set_drvdata(monreader_device, NULL); | ||
| 360 | 361 | ||
| 361 | for (i = 0; i < MON_MSGLIM; i++) | 362 | for (i = 0; i < MON_MSGLIM; i++) |
| 362 | kfree(monpriv->msg_array[i]); | 363 | kfree(monpriv->msg_array[i]); |
diff --git a/drivers/s390/char/sclp_quiesce.c b/drivers/s390/char/sclp_quiesce.c index 84c191c1cd62..05909a7df8b3 100644 --- a/drivers/s390/char/sclp_quiesce.c +++ b/drivers/s390/char/sclp_quiesce.c | |||
| @@ -20,9 +20,12 @@ | |||
| 20 | 20 | ||
| 21 | #include "sclp.h" | 21 | #include "sclp.h" |
| 22 | 22 | ||
| 23 | static void (*old_machine_restart)(char *); | ||
| 24 | static void (*old_machine_halt)(void); | ||
| 25 | static void (*old_machine_power_off)(void); | ||
| 26 | |||
| 23 | /* Shutdown handler. Signal completion of shutdown by loading special PSW. */ | 27 | /* Shutdown handler. Signal completion of shutdown by loading special PSW. */ |
| 24 | static void | 28 | static void do_machine_quiesce(void) |
| 25 | do_machine_quiesce(void) | ||
| 26 | { | 29 | { |
| 27 | psw_t quiesce_psw; | 30 | psw_t quiesce_psw; |
| 28 | 31 | ||
| @@ -33,23 +36,48 @@ do_machine_quiesce(void) | |||
| 33 | } | 36 | } |
| 34 | 37 | ||
| 35 | /* Handler for quiesce event. Start shutdown procedure. */ | 38 | /* Handler for quiesce event. Start shutdown procedure. */ |
| 36 | static void | 39 | static void sclp_quiesce_handler(struct evbuf_header *evbuf) |
| 37 | sclp_quiesce_handler(struct evbuf_header *evbuf) | ||
| 38 | { | 40 | { |
| 39 | _machine_restart = (void *) do_machine_quiesce; | 41 | if (_machine_restart != (void *) do_machine_quiesce) { |
| 40 | _machine_halt = do_machine_quiesce; | 42 | old_machine_restart = _machine_restart; |
| 41 | _machine_power_off = do_machine_quiesce; | 43 | old_machine_halt = _machine_halt; |
| 44 | old_machine_power_off = _machine_power_off; | ||
| 45 | _machine_restart = (void *) do_machine_quiesce; | ||
| 46 | _machine_halt = do_machine_quiesce; | ||
| 47 | _machine_power_off = do_machine_quiesce; | ||
| 48 | } | ||
| 42 | ctrl_alt_del(); | 49 | ctrl_alt_del(); |
| 43 | } | 50 | } |
| 44 | 51 | ||
| 52 | /* Undo machine restart/halt/power_off modification on resume */ | ||
| 53 | static void sclp_quiesce_pm_event(struct sclp_register *reg, | ||
| 54 | enum sclp_pm_event sclp_pm_event) | ||
| 55 | { | ||
| 56 | switch (sclp_pm_event) { | ||
| 57 | case SCLP_PM_EVENT_RESTORE: | ||
| 58 | if (old_machine_restart) { | ||
| 59 | _machine_restart = old_machine_restart; | ||
| 60 | _machine_halt = old_machine_halt; | ||
| 61 | _machine_power_off = old_machine_power_off; | ||
| 62 | old_machine_restart = NULL; | ||
| 63 | old_machine_halt = NULL; | ||
| 64 | old_machine_power_off = NULL; | ||
| 65 | } | ||
| 66 | break; | ||
| 67 | case SCLP_PM_EVENT_FREEZE: | ||
| 68 | case SCLP_PM_EVENT_THAW: | ||
| 69 | break; | ||
| 70 | } | ||
| 71 | } | ||
| 72 | |||
| 45 | static struct sclp_register sclp_quiesce_event = { | 73 | static struct sclp_register sclp_quiesce_event = { |
| 46 | .receive_mask = EVTYP_SIGQUIESCE_MASK, | 74 | .receive_mask = EVTYP_SIGQUIESCE_MASK, |
| 47 | .receiver_fn = sclp_quiesce_handler | 75 | .receiver_fn = sclp_quiesce_handler, |
| 76 | .pm_event_fn = sclp_quiesce_pm_event | ||
| 48 | }; | 77 | }; |
| 49 | 78 | ||
| 50 | /* Initialize quiesce driver. */ | 79 | /* Initialize quiesce driver. */ |
| 51 | static int __init | 80 | static int __init sclp_quiesce_init(void) |
| 52 | sclp_quiesce_init(void) | ||
| 53 | { | 81 | { |
| 54 | return sclp_register(&sclp_quiesce_event); | 82 | return sclp_register(&sclp_quiesce_event); |
| 55 | } | 83 | } |
