• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1  // SPDX-License-Identifier: GPL-2.0-or-later
2  
3  /*
4   *
5   * Copyright (C) IBM Corporation, 2004
6   *
7   * Author: Max Asböck <amax@us.ibm.com>
8   */
9  
10  #include <linux/sched/signal.h>
11  #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)
22  static struct {
23  	struct dot_command_header	header;
24  	unsigned char			command[3];
25  } rhb_dot_cmd = {
26  	.header = {
27  		.type =		sp_read,
28  		.command_size = 3,
29  		.data_size =	0,
30  		.status =	0
31  	},
32  	.command = { 4, 3, 6 }
33  };
34  #pragma pack()
35  
ibmasm_init_reverse_heartbeat(struct service_processor * sp,struct reverse_heartbeat * rhb)36  void 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   */
ibmasm_start_reverse_heartbeat(struct service_processor * sp,struct reverse_heartbeat * rhb)48  int 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  
54  	cmd = ibmasm_new_command(sp, sizeof rhb_dot_cmd);
55  	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  
67  		wait_event_interruptible_timeout(rhb->wait,
68  			rhb->stopped,
69  			REVERSE_HEARTBEAT_TIMEOUT * HZ);
70  
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  
ibmasm_stop_reverse_heartbeat(struct reverse_heartbeat * rhb)82  void ibmasm_stop_reverse_heartbeat(struct reverse_heartbeat *rhb)
83  {
84  	rhb->stopped = 1;
85  	wake_up_interruptible(&rhb->wait);
86  }
87