Blame lib/ipmi_quantaoem.c

Packit Service 8d9f00
/*
Packit Service 8d9f00
 * Copyright (c) 2018 Quanta Computer Inc. All rights reserved.
Packit Service 8d9f00
 *
Packit Service 8d9f00
 * Redistribution and use in source and binary forms, with or without
Packit Service 8d9f00
 * modification, are permitted provided that the following conditions
Packit Service 8d9f00
 * are met:
Packit Service 8d9f00
 *
Packit Service 8d9f00
 * Redistribution of source code must retain the above copyright
Packit Service 8d9f00
 * notice, this list of conditions and the following disclaimer.
Packit Service 8d9f00
 *
Packit Service 8d9f00
 * Redistribution in binary form must reproduce the above copyright
Packit Service 8d9f00
 * notice, this list of conditions and the following disclaimer in the
Packit Service 8d9f00
 * documentation and/or other materials provided with the distribution.
Packit Service 8d9f00
 *
Packit Service 8d9f00
 * Neither the name of Quanta Computer Inc. or the names of
Packit Service 8d9f00
 * contributors may be used to endorse or promote products derived
Packit Service 8d9f00
 * from this software without specific prior written permission.
Packit Service 8d9f00
 *
Packit Service 8d9f00
 * This software is provided "AS IS," without a warranty of any kind.
Packit Service 8d9f00
 * ALL EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES,
Packit Service 8d9f00
 * INCLUDING ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A
Packit Service 8d9f00
 * PARTICULAR PURPOSE OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED.
Packit Service 8d9f00
 * Quanta Computer Inc. AND ITS LICENSORS SHALL NOT BE LIABLE
Packit Service 8d9f00
 * FOR ANY DAMAGES SUFFERED BY LICENSEE AS A RESULT OF USING, MODIFYING
Packit Service 8d9f00
 * OR DISTRIBUTING THIS SOFTWARE OR ITS DERIVATIVES.  IN NO EVENT WILL
Packit Service 8d9f00
 * Quanta Computer Inc. OR ITS LICENSORS BE LIABLE FOR ANY LOST REVENUE, PROFIT OR DATA,
Packit Service 8d9f00
 * OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, INCIDENTAL OR
Packit Service 8d9f00
 * PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY OF
Packit Service 8d9f00
 * LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE,
Packit Service 8d9f00
 * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
Packit Service 8d9f00
 */
Packit Service 8d9f00
#define _XOPEN_SOURCE
Packit Service 8d9f00
Packit Service 8d9f00
#include <stdlib.h>
Packit Service 8d9f00
#include <stdio.h>
Packit Service 8d9f00
#include <string.h>
Packit Service 8d9f00
#include <strings.h>
Packit Service 8d9f00
#include <sys/socket.h>
Packit Service 8d9f00
#include <netinet/in.h>
Packit Service 8d9f00
#include <arpa/inet.h>
Packit Service 8d9f00
#include <errno.h>
Packit Service 8d9f00
#include <time.h>
Packit Service 8d9f00
#include <unistd.h>
Packit Service 8d9f00
#include <signal.h>
Packit Service 8d9f00
#include <ctype.h>
Packit Service 8d9f00
#include <sys/time.h>
Packit Service 8d9f00
#include <limits.h>
Packit Service 8d9f00
#include <fcntl.h>
Packit Service 8d9f00
#include <sys/select.h>
Packit Service 8d9f00
#include <termios.h>
Packit Service 8d9f00
#include <ipmitool/ipmi.h>
Packit Service 8d9f00
#include <ipmitool/ipmi_mc.h>
Packit Service 8d9f00
#include <ipmitool/ipmi_intf.h>
Packit Service 8d9f00
#include <ipmitool/helper.h>
Packit Service 8d9f00
#include <ipmitool/log.h>
Packit Service 8d9f00
#include <ipmitool/ipmi_sel.h>
Packit Service 8d9f00
#include <ipmitool/ipmi_sdr.h>
Packit Service 8d9f00
#include <ipmitool/ipmi_strings.h>
Packit Service 8d9f00
#include <ipmitool/ipmi_channel.h>
Packit Service 8d9f00
#include <ipmitool/ipmi_quantaoem.h>
Packit Service 8d9f00
#include <ipmitool/ipmi_raw.h>
Packit Service 8d9f00
Packit Service 8d9f00
/* Max Size of the description String to be displyed for the Each sel entry */
Packit Service 8d9f00
#define	SIZE_OF_DESC 128
Packit Service 8d9f00
Packit Service 8d9f00
#define CPU_SHIFT 6
Packit Service 8d9f00
#define CPU_MASK 0X03
Packit Service 8d9f00
#define CPU_NUM(x) (((x) >> CPU_SHIFT) & CPU_MASK)
Packit Service 8d9f00
Packit Service 8d9f00
#define CHANNEL_BASE 0x41
Packit Service 8d9f00
#define CHANNEL_SHIFT 3
Packit Service 8d9f00
#define CHANNEL_MASK 0x07
Packit Service 8d9f00
#define CHANNEL_OFFSET(x) (((x) >> CHANNEL_SHIFT) & CHANNEL_MASK)
Packit Service 8d9f00
#define CHANNEL_NUM(x) (CHANNEL_BASE + CHANNEL_OFFSET(x))
Packit Service 8d9f00
Packit Service 8d9f00
#define DIMM_MASK 0x07
Packit Service 8d9f00
#define DIMM_NUM(x) ((x) & DIMM_MASK)
Packit Service 8d9f00
Packit Service 8d9f00
#define	GET_PLATFORM_ID_DATA_SIZE 4
Packit Service 8d9f00
Packit Service 8d9f00
// Magic code to check if it's valid command
Packit Service 8d9f00
#define QCT_MAGIC_1 0x4C
Packit Service 8d9f00
#define QCT_MAGIC_2 0x1C
Packit Service 8d9f00
#define QCT_MAGIC_3 0x00
Packit Service 8d9f00
#define QCT_MAGIC_4 0x02
Packit Service 8d9f00
Packit Service 8d9f00
qct_platform_t
Packit Service 8d9f00
oem_qct_get_platform_id(struct ipmi_intf *intf)
Packit Service 8d9f00
{
Packit Service 8d9f00
	/* Execute a Get platform ID command to determine the board */
Packit Service 8d9f00
	struct ipmi_rs *rsp;
Packit Service 8d9f00
	struct ipmi_rq req;
Packit Service 8d9f00
	qct_platform_t platform_id;
Packit Service 8d9f00
	uint8_t msg_data[GET_PLATFORM_ID_DATA_SIZE];
Packit Service 8d9f00
Packit Service 8d9f00
	/* Ask for IPMI v2 data as well */
Packit Service 8d9f00
	msg_data[0] = QCT_MAGIC_1;
Packit Service 8d9f00
	msg_data[1] = QCT_MAGIC_2;
Packit Service 8d9f00
	msg_data[2] = QCT_MAGIC_3;
Packit Service 8d9f00
	msg_data[3] = QCT_MAGIC_4;
Packit Service 8d9f00
Packit Service 8d9f00
	memset(&req, 0, sizeof(req));
Packit Service 8d9f00
	req.msg.netfn = OEM_QCT_NETFN;
Packit Service 8d9f00
	req.msg.cmd = OEM_QCT_GET_INFO;
Packit Service 8d9f00
	req.msg.data = msg_data;
Packit Service 8d9f00
	req.msg.data_len = sizeof(msg_data);
Packit Service 8d9f00
Packit Service 8d9f00
	rsp = intf->sendrecv(intf, &req;;
Packit Service 8d9f00
	if (rsp == NULL) {
Packit Service 8d9f00
		lprintf(LOG_ERR, "Get Platform ID command failed");
Packit Service 8d9f00
		return 0;
Packit Service 8d9f00
	}
Packit Service 8d9f00
	if (rsp->ccode) {
Packit Service 8d9f00
		lprintf(LOG_ERR, "Get Platform ID command failed: %#x %s",
Packit Service 8d9f00
		        rsp->ccode, val2str(rsp->ccode, completion_code_vals));
Packit Service 8d9f00
		return 0;
Packit Service 8d9f00
	}
Packit Service 8d9f00
	platform_id = rsp->data[0];
Packit Service 8d9f00
	lprintf(LOG_DEBUG,"Platform ID: %hhx", rsp->data[0]);
Packit Service 8d9f00
	return platform_id;
Packit Service 8d9f00
}
Packit Service 8d9f00
Packit Service 8d9f00
char *
Packit Service 8d9f00
oem_qct_get_evt_desc(struct ipmi_intf *intf, struct sel_event_record *rec)
Packit Service 8d9f00
{
Packit Service 8d9f00
	struct ipmi_rs *rsp;
Packit Service 8d9f00
	struct ipmi_rq req;
Packit Service 8d9f00
	char *desc = NULL;
Packit Service 8d9f00
	int data;
Packit Service 8d9f00
	int sensor_type;
Packit Service 8d9f00
	qct_platform_t platform_id;
Packit Service 8d9f00
Packit Service 8d9f00
	/* Get the OEM event Bytes of the SEL Records byte 15 to data */
Packit Service 8d9f00
	data = rec->sel_type.standard_type.event_data[2];
Packit Service 8d9f00
	/* Check for the Standard Event type == 0x6F */
Packit Service 8d9f00
	if (rec->sel_type.standard_type.event_type != 0x6F) {
Packit Service 8d9f00
		goto out;
Packit Service 8d9f00
	}
Packit Service 8d9f00
	/* Allocate mem for te Description string */
Packit Service 8d9f00
	desc = malloc(SIZE_OF_DESC);
Packit Service 8d9f00
	if (desc == NULL) {
Packit Service 8d9f00
		lprintf(LOG_ERR, "ipmitool: malloc failure");
Packit Service 8d9f00
		goto out;
Packit Service 8d9f00
	}
Packit Service 8d9f00
	memset(desc, 0, SIZE_OF_DESC);
Packit Service 8d9f00
	sensor_type = rec->sel_type.standard_type.sensor_type;
Packit Service 8d9f00
	switch (sensor_type) {
Packit Service 8d9f00
	case SENSOR_TYPE_MEMORY:
Packit Service 8d9f00
		memset(&req, 0, sizeof (req));
Packit Service 8d9f00
		req.msg.netfn = IPMI_NETFN_APP;
Packit Service 8d9f00
		req.msg.lun = 0;
Packit Service 8d9f00
		req.msg.cmd = BMC_GET_DEVICE_ID;
Packit Service 8d9f00
		req.msg.data = NULL;
Packit Service 8d9f00
		req.msg.data_len = 0;
Packit Service 8d9f00
Packit Service 8d9f00
		rsp = intf->sendrecv(intf, &req;;
Packit Service 8d9f00
		if (rsp == NULL) {
Packit Service 8d9f00
			lprintf(LOG_ERR, " Error getting system info");
Packit Service 8d9f00
			goto out;
Packit Service 8d9f00
		} else if (rsp->ccode) {
Packit Service 8d9f00
			lprintf(LOG_ERR, " Error getting system info: %s",
Packit Service 8d9f00
			        val2str(rsp->ccode, completion_code_vals));
Packit Service 8d9f00
			goto out;
Packit Service 8d9f00
		}
Packit Service 8d9f00
		/* check the platform type */
Packit Service 8d9f00
		platform_id = oem_qct_get_platform_id(intf);
Packit Service 8d9f00
		if (OEM_QCT_PLATFORM_PURLEY == platform_id) {
Packit Service 8d9f00
			snprintf(desc, SIZE_OF_DESC, "CPU%d_%c%d",
Packit Service 8d9f00
			         CPU_NUM(data),
Packit Service 8d9f00
			         CHANNEL_NUM(data),
Packit Service 8d9f00
			         DIMM_NUM(data));
Packit Service 8d9f00
		}
Packit Service 8d9f00
		break;
Packit Service 8d9f00
	default:
Packit Service 8d9f00
		goto out;
Packit Service 8d9f00
	}
Packit Service 8d9f00
	return desc;
Packit Service 8d9f00
out:
Packit Service 8d9f00
	if (desc) {
Packit Service 8d9f00
		free(desc);
Packit Service 8d9f00
		desc = NULL;
Packit Service 8d9f00
	}
Packit Service 8d9f00
	return desc;
Packit Service 8d9f00
}