blob: 6567df638ea93e0e78aed51fe2cd67a040c4bc95 [file] [log] [blame]
Thomas Gleixner1a59d1b82019-05-27 08:55:05 +02001// SPDX-License-Identifier: GPL-2.0-or-later
Linus Torvalds1da177e2005-04-16 15:20:36 -07002
3/*
Linus Torvalds1da177e2005-04-16 15:20:36 -07004 *
5 * Copyright (C) IBM Corporation, 2004
6 *
Al Virod36b6912011-12-29 17:09:01 -05007 * Author: Max Asböck <amax@us.ibm.com>
Linus Torvalds1da177e2005-04-16 15:20:36 -07008 */
9
Ingo Molnar174cd4b2017-02-02 19:15:33 +010010#include <linux/sched/signal.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070011#include "ibmasm.h"
12#include "dot_command.h"
13
14/*
15 * Reverse Heartbeat, i.e. heartbeats sent from the driver to the
16 * service processor.
17 * These heartbeats are initiated by user level programs.
18 */
19
20/* the reverse heartbeat dot command */
21#pragma pack(1)
22static struct {
23 struct dot_command_header header;
24 unsigned char command[3];
25} rhb_dot_cmd = {
26 .header = {
Dmitry Torokhov3110dc72007-07-17 04:03:58 -070027 .type = sp_read,
Linus Torvalds1da177e2005-04-16 15:20:36 -070028 .command_size = 3,
29 .data_size = 0,
Dmitry Torokhov3110dc72007-07-17 04:03:58 -070030 .status = 0
Linus Torvalds1da177e2005-04-16 15:20:36 -070031 },
32 .command = { 4, 3, 6 }
33};
34#pragma pack()
35
36void ibmasm_init_reverse_heartbeat(struct service_processor *sp, struct reverse_heartbeat *rhb)
37{
38 init_waitqueue_head(&rhb->wait);
39 rhb->stopped = 0;
40}
41
42/**
43 * start_reverse_heartbeat
44 * Loop forever, sending a reverse heartbeat dot command to the service
45 * processor, then sleeping. The loop comes to an end if the service
46 * processor fails to respond 3 times or we were interrupted.
47 */
48int ibmasm_start_reverse_heartbeat(struct service_processor *sp, struct reverse_heartbeat *rhb)
49{
50 struct command *cmd;
51 int times_failed = 0;
52 int result = 1;
53
Max Asbock88187602005-06-21 17:16:36 -070054 cmd = ibmasm_new_command(sp, sizeof rhb_dot_cmd);
Linus Torvalds1da177e2005-04-16 15:20:36 -070055 if (!cmd)
56 return -ENOMEM;
57
58 while (times_failed < 3) {
59 memcpy(cmd->buffer, (void *)&rhb_dot_cmd, sizeof rhb_dot_cmd);
60 cmd->status = IBMASM_CMD_PENDING;
61 ibmasm_exec_command(sp, cmd);
62 ibmasm_wait_for_response(cmd, IBMASM_CMD_TIMEOUT_NORMAL);
63
64 if (cmd->status != IBMASM_CMD_COMPLETE)
65 times_failed++;
66
Dmitry Torokhov3110dc72007-07-17 04:03:58 -070067 wait_event_interruptible_timeout(rhb->wait,
Linus Torvalds1da177e2005-04-16 15:20:36 -070068 rhb->stopped,
Dmitry Torokhov3110dc72007-07-17 04:03:58 -070069 REVERSE_HEARTBEAT_TIMEOUT * HZ);
Linus Torvalds1da177e2005-04-16 15:20:36 -070070
71 if (signal_pending(current) || rhb->stopped) {
72 result = -EINTR;
73 break;
74 }
75 }
76 command_put(cmd);
77 rhb->stopped = 0;
78
79 return result;
80}
81
82void ibmasm_stop_reverse_heartbeat(struct reverse_heartbeat *rhb)
83{
84 rhb->stopped = 1;
85 wake_up_interruptible(&rhb->wait);
86}