ipmi_smi.h 7.8 KB
Newer Older
1
/* SPDX-License-Identifier: GPL-2.0+ */
L
Linus Torvalds 已提交
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
/*
 * ipmi_smi.h
 *
 * MontaVista IPMI system management interface
 *
 * Author: MontaVista Software, Inc.
 *         Corey Minyard <minyard@mvista.com>
 *         source@mvista.com
 *
 * Copyright 2002 MontaVista Software Inc.
 *
 */

#ifndef __LINUX_IPMI_SMI_H
#define __LINUX_IPMI_SMI_H

#include <linux/ipmi_msgdefs.h>
#include <linux/proc_fs.h>
20
#include <linux/platform_device.h>
21
#include <linux/ipmi.h>
L
Linus Torvalds 已提交
22

23 24
struct device;

25 26 27 28
/*
 * This files describes the interface for IPMI system management interface
 * drivers to bind into the IPMI message handler.
 */
L
Linus Torvalds 已提交
29 30

/* Structure for the low-level drivers. */
31
struct ipmi_smi;
L
Linus Torvalds 已提交
32

33 34
/*
 * Flags for set_check_watch() below.  Tells if the SMI should be
35
 * waiting for watchdog timeouts, commands and/or messages.
36
 */
37 38 39
#define IPMI_WATCH_MASK_CHECK_MESSAGES	(1 << 0)
#define IPMI_WATCH_MASK_CHECK_WATCHDOG	(1 << 1)
#define IPMI_WATCH_MASK_CHECK_COMMANDS	(1 << 2)
40

L
Linus Torvalds 已提交
41 42 43 44 45 46 47 48 49 50 51 52 53
/*
 * Messages to/from the lower layer.  The smi interface will take one
 * of these to send. After the send has occurred and a response has
 * been received, it will report this same data structure back up to
 * the upper layer.  If an error occurs, it should fill in the
 * response with an error code in the completion code location. When
 * asynchronous data is received, one of these is allocated, the
 * data_size is set to zero and the response holds the data from the
 * get message or get event command that the interface initiated.
 * Note that it is the interfaces responsibility to detect
 * asynchronous data and messages and request them from the
 * interface.
 */
54
struct ipmi_smi_msg {
L
Linus Torvalds 已提交
55 56 57 58 59 60 61 62 63 64 65
	struct list_head link;

	long    msgid;
	void    *user_data;

	int           data_size;
	unsigned char data[IPMI_MAX_MSG_LENGTH];

	int           rsp_size;
	unsigned char rsp[IPMI_MAX_MSG_LENGTH];

66 67 68 69
	/*
	 * Will be called when the system is done with the message
	 * (presumably to free it).
	 */
L
Linus Torvalds 已提交
70 71 72
	void (*done)(struct ipmi_smi_msg *msg);
};

73
struct ipmi_smi_handlers {
L
Linus Torvalds 已提交
74 75
	struct module *owner;

76 77 78 79 80 81
	/*
	 * The low-level interface cannot start sending messages to
	 * the upper layer until this function is called.  This may
	 * not be NULL, the lower layer must take the interface from
	 * this call.
	 */
82 83
	int (*start_processing)(void            *send_info,
				struct ipmi_smi *new_intf);
84

85 86 87 88 89 90
	/*
	 * When called, the low-level interface should disable all
	 * processing, it should be complete shut down when it returns.
	 */
	void (*shutdown)(void *send_info);

91 92 93 94 95 96 97
	/*
	 * Get the detailed private info of the low level interface and store
	 * it into the structure of ipmi_smi_data. For example: the
	 * ACPI device handle will be returned for the pnp_acpi IPMI device.
	 */
	int (*get_smi_info)(void *send_info, struct ipmi_smi_info *data);

98 99 100 101 102 103 104 105 106
	/*
	 * Called to enqueue an SMI message to be sent.  This
	 * operation is not allowed to fail.  If an error occurs, it
	 * should report back the error in a received message.  It may
	 * do this in the current call context, since no write locks
	 * are held when this is run.  Message are delivered one at
	 * a time by the message handler, a new message will not be
	 * delivered until the previous message is returned.
	 */
L
Linus Torvalds 已提交
107
	void (*sender)(void                *send_info,
108
		       struct ipmi_smi_msg *msg);
L
Linus Torvalds 已提交
109

110 111 112 113
	/*
	 * Called by the upper layer to request that we try to get
	 * events from the BMC we are attached to.
	 */
L
Linus Torvalds 已提交
114 115
	void (*request_events)(void *send_info);

116 117
	/*
	 * Called by the upper layer when some user requires that the
118 119 120 121 122 123 124
	 * interface watch for received messages and watchdog
	 * pretimeouts (basically do a "Get Flags", or not.  Used by
	 * the SMI to know if it should watch for these.  This may be
	 * NULL if the SMI does not implement it.  watch_mask is from
	 * IPMI_WATCH_MASK_xxx above.  The interface should run slower
	 * timeouts for just watchdog checking or faster timeouts when
	 * waiting for the message queue.
125
	 */
126
	void (*set_need_watch)(void *send_info, unsigned int watch_mask);
127

128 129 130 131 132
	/*
	 * Called when flushing all pending messages.
	 */
	void (*flush_messages)(void *send_info);

133 134 135 136 137 138 139
	/*
	 * Called when the interface should go into "run to
	 * completion" mode.  If this call sets the value to true, the
	 * interface should make sure that all messages are flushed
	 * out and that none are pending, and any new requests are run
	 * to completion immediately.
	 */
C
Corey Minyard 已提交
140
	void (*set_run_to_completion)(void *send_info, bool run_to_completion);
L
Linus Torvalds 已提交
141

142 143 144 145
	/*
	 * Called to poll for work to do.  This is so upper layers can
	 * poll for operations during things like crash dumps.
	 */
L
Linus Torvalds 已提交
146 147
	void (*poll)(void *send_info);

148 149 150 151 152 153 154
	/*
	 * Enable/disable firmware maintenance mode.  Note that this
	 * is *not* the modes defined, this is simply an on/off
	 * setting.  The message handler does the mode handling.  Note
	 * that this is called from interrupt context, so it cannot
	 * block.
	 */
C
Corey Minyard 已提交
155
	void (*set_maintenance_mode)(void *send_info, bool enable);
L
Linus Torvalds 已提交
156 157
};

158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173
struct ipmi_device_id {
	unsigned char device_id;
	unsigned char device_revision;
	unsigned char firmware_revision_1;
	unsigned char firmware_revision_2;
	unsigned char ipmi_version;
	unsigned char additional_device_support;
	unsigned int  manufacturer_id;
	unsigned int  product_id;
	unsigned char aux_firmware_revision[4];
	unsigned int  aux_firmware_revision_set : 1;
};

#define ipmi_version_major(v) ((v)->ipmi_version & 0xf)
#define ipmi_version_minor(v) ((v)->ipmi_version >> 4)

174 175
/*
 * Take a pointer to an IPMI response and extract device id information from
176 177 178 179 180
 * it. @netfn is in the IPMI_NETFN_ format, so may need to be shifted from
 * a SI response.
 */
static inline int ipmi_demangle_device_id(uint8_t netfn, uint8_t cmd,
					  const unsigned char *data,
C
Corey Minyard 已提交
181 182
					  unsigned int data_len,
					  struct ipmi_device_id *id)
183
{
184
	if (data_len < 7)
C
Corey Minyard 已提交
185
		return -EINVAL;
186
	if (netfn != IPMI_NETFN_APP_RESPONSE || cmd != IPMI_GET_DEVICE_ID_CMD)
C
Corey Minyard 已提交
187 188
		/* Strange, didn't get the response we expected. */
		return -EINVAL;
189
	if (data[0] != 0)
C
Corey Minyard 已提交
190 191 192
		/* That's odd, it shouldn't be able to fail. */
		return -EINVAL;

193 194 195
	data++;
	data_len--;

196 197 198 199 200 201
	id->device_id = data[0];
	id->device_revision = data[1];
	id->firmware_revision_1 = data[2];
	id->firmware_revision_2 = data[3];
	id->ipmi_version = data[4];
	id->additional_device_support = data[5];
202
	if (data_len >= 11) {
C
Corey Minyard 已提交
203 204 205 206 207 208 209
		id->manufacturer_id = (data[6] | (data[7] << 8) |
				       (data[8] << 16));
		id->product_id = data[9] | (data[10] << 8);
	} else {
		id->manufacturer_id = 0;
		id->product_id = 0;
	}
210 211 212 213 214
	if (data_len >= 15) {
		memcpy(id->aux_firmware_revision, data+11, 4);
		id->aux_firmware_revision_set = 1;
	} else
		id->aux_firmware_revision_set = 0;
C
Corey Minyard 已提交
215 216

	return 0;
217 218
}

219 220 221 222 223 224 225 226
/*
 * Add a low-level interface to the IPMI driver.  Note that if the
 * interface doesn't know its slave address, it should pass in zero.
 * The low-level interface should not deliver any messages to the
 * upper layer until the start_processing() function in the handlers
 * is called, and the lower layer must get the interface from that
 * call.
 */
227
int ipmi_register_smi(const struct ipmi_smi_handlers *handlers,
L
Linus Torvalds 已提交
228
		      void                     *send_info,
229
		      struct device            *dev,
230
		      unsigned char            slave_addr);
L
Linus Torvalds 已提交
231 232 233 234 235

/*
 * Remove a low-level interface from the IPMI driver.  This will
 * return an error if the interface is still in use by a user.
 */
236
void ipmi_unregister_smi(struct ipmi_smi *intf);
L
Linus Torvalds 已提交
237 238 239

/*
 * The lower layer reports received messages through this interface.
240
 * The data_size should be zero if this is an asynchronous message.  If
L
Linus Torvalds 已提交
241 242 243
 * the lower layer gets an error sending a message, it should format
 * an error response in the message response.
 */
244
void ipmi_smi_msg_received(struct ipmi_smi     *intf,
L
Linus Torvalds 已提交
245 246 247
			   struct ipmi_smi_msg *msg);

/* The lower layer received a watchdog pre-timeout on interface. */
248
void ipmi_smi_watchdog_pretimeout(struct ipmi_smi *intf);
L
Linus Torvalds 已提交
249 250 251 252 253 254 255 256

struct ipmi_smi_msg *ipmi_alloc_smi_msg(void);
static inline void ipmi_free_smi_msg(struct ipmi_smi_msg *msg)
{
	msg->done(msg);
}

#endif /* __LINUX_IPMI_SMI_H */