Files
OpenCellular/driver/temp_sensor/g78x.c
Kevin K Wong bf24d5b264 temp sensor: add support for G782
modify g781.c/.h to g78x.c/.h to suppor both G781/G782 temp sensor
based on CONFIG_TEMP_SENSOR_G781 or CONFIG_TEMP_SENSOR_G782

BUG=none
BRANCH=none
TEST=make buildall; able to get temperature data on board with G782

Change-Id: Ia32c85e9964bfd7c0c5263f04368bc001a27fe10
Signed-off-by: Kevin K Wong <kevin.k.wong@intel.com>
Reviewed-on: https://chromium-review.googlesource.com/334228
Reviewed-by: Shawn N <shawnn@chromium.org>
2016-04-03 10:27:40 -07:00

236 lines
5.1 KiB
C

/* Copyright 2016 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
/* G781/G782 temperature sensor module for Chrome EC */
#include "common.h"
#include "console.h"
#include "g78x.h"
#include "gpio.h"
#include "i2c.h"
#include "hooks.h"
#include "util.h"
static int temp_val_local;
static int temp_val_remote1;
#ifdef CONFIG_TEMP_SENSOR_G782
static int temp_val_remote2;
#endif
/**
* Determine whether the sensor is powered.
*
* @return non-zero the g781/g782 sensor is powered.
*/
static int has_power(void)
{
#ifdef CONFIG_TEMP_SENSOR_POWER_GPIO
return gpio_get_level(CONFIG_TEMP_SENSOR_POWER_GPIO);
#else
return 1;
#endif
}
static int raw_read8(const int offset, int *data_ptr)
{
return i2c_read8(I2C_PORT_THERMAL, G78X_I2C_ADDR, offset, data_ptr);
}
#ifdef CONFIG_CMD_TEMP_SENSOR
static int raw_write8(const int offset, int data)
{
return i2c_write8(I2C_PORT_THERMAL, G78X_I2C_ADDR, offset, data);
}
#endif
static int get_temp(const int offset, int *temp_ptr)
{
int rv;
int temp_raw = 0;
rv = raw_read8(offset, &temp_raw);
if (rv < 0)
return rv;
*temp_ptr = (int)(int8_t)temp_raw;
return EC_SUCCESS;
}
#ifdef CONFIG_CMD_TEMP_SENSOR
static int set_temp(const int offset, int temp)
{
if (temp < -127 || temp > 127)
return EC_ERROR_INVAL;
return raw_write8(offset, (uint8_t)temp);
}
#endif
int g78x_get_val(int idx, int *temp_ptr)
{
if (!has_power())
return EC_ERROR_NOT_POWERED;
switch (idx) {
case G78X_IDX_INTERNAL:
*temp_ptr = temp_val_local;
break;
case G78X_IDX_EXTERNAL1:
*temp_ptr = temp_val_remote1;
break;
#ifdef CONFIG_TEMP_SENSOR_G782
case G78X_IDX_EXTERNAL2:
*temp_ptr = temp_val_remote2;
break;
#endif
default:
return EC_ERROR_UNKNOWN;
}
return EC_SUCCESS;
}
static void temp_sensor_poll(void)
{
if (!has_power())
return;
get_temp(G78X_TEMP_LOCAL, &temp_val_local);
temp_val_local = C_TO_K(temp_val_local);
get_temp(G78X_TEMP_REMOTE1, &temp_val_remote1);
temp_val_remote1 = C_TO_K(temp_val_remote1);
#ifdef CONFIG_TEMP_SENSOR_G782
get_temp(G78X_TEMP_REMOTE2, &temp_val_remote2);
temp_val_remote2 = C_TO_K(temp_val_remote2);
#endif
}
DECLARE_HOOK(HOOK_SECOND, temp_sensor_poll, HOOK_PRIO_TEMP_SENSOR);
#ifdef CONFIG_CMD_TEMP_SENSOR
static void print_temps(const char *name,
const int temp_reg,
const int therm_limit_reg,
const int high_limit_reg,
const int low_limit_reg)
{
int value;
ccprintf("%s:\n", name);
if (get_temp(temp_reg, &value) == EC_SUCCESS)
ccprintf(" Temp: %3dC\n", value);
if (get_temp(therm_limit_reg, &value) == EC_SUCCESS)
ccprintf(" Therm Trip: %3dC\n", value);
if (get_temp(high_limit_reg, &value) == EC_SUCCESS)
ccprintf(" High Alarm: %3dC\n", value);
if (get_temp(low_limit_reg, &value) == EC_SUCCESS)
ccprintf(" Low Alarm: %3dC\n", value);
}
static int print_status(void)
{
int value;
if (!has_power()) {
ccprintf("ERROR: Temp sensor not powered.\n");
return EC_ERROR_NOT_POWERED;
}
print_temps("Local", G78X_TEMP_LOCAL,
G78X_LOCAL_TEMP_THERM_LIMIT,
G78X_LOCAL_TEMP_HIGH_LIMIT_R,
G78X_LOCAL_TEMP_LOW_LIMIT_R);
print_temps("Remote1", G78X_TEMP_REMOTE1,
G78X_REMOTE1_TEMP_THERM_LIMIT,
G78X_REMOTE1_TEMP_HIGH_LIMIT_R,
G78X_REMOTE1_TEMP_LOW_LIMIT_R);
#ifdef CONFIG_TEMP_SENSOR_G782
print_temps("Remote2", G78X_TEMP_REMOTE1,
G78X_REMOTE2_TEMP_THERM_LIMIT,
G78X_REMOTE2_TEMP_HIGH_LIMIT_R,
G78X_REMOTE2_TEMP_LOW_LIMIT_R);
#endif
ccprintf("\n");
if (raw_read8(G78X_STATUS, &value) == EC_SUCCESS)
ccprintf("STATUS: %08b\n", value);
#ifdef CONFIG_TEMP_SENSOR_G782
if (raw_read8(G78X_STATUS1, &value) == EC_SUCCESS)
ccprintf("STATUS1: %08b\n", value);
#endif
if (raw_read8(G78X_CONFIGURATION_R, &value) == EC_SUCCESS)
ccprintf("CONFIG: %08b\n", value);
return EC_SUCCESS;
}
static int command_g78x(int argc, char **argv)
{
char *command;
char *e;
int data;
int offset;
int rv;
if (!has_power()) {
ccprintf("ERROR: Temp sensor not powered.\n");
return EC_ERROR_NOT_POWERED;
}
/* If no args just print status */
if (argc == 1)
return print_status();
if (argc < 3)
return EC_ERROR_PARAM_COUNT;
command = argv[1];
offset = strtoi(argv[2], &e, 0);
if (*e || offset < 0 || offset > 255)
return EC_ERROR_PARAM2;
if (!strcasecmp(command, "getbyte")) {
rv = raw_read8(offset, &data);
if (rv < 0)
return rv;
ccprintf("Byte at offset 0x%02x is %08b\n", offset, data);
return rv;
}
/* Remaining commands are of the form "g78x set-command offset data" */
if (argc != 4)
return EC_ERROR_PARAM_COUNT;
data = strtoi(argv[3], &e, 0);
if (*e)
return EC_ERROR_PARAM3;
if (!strcasecmp(command, "settemp")) {
ccprintf("Setting 0x%02x to %dC\n", offset, data);
rv = set_temp(offset, data);
} else if (!strcasecmp(command, "setbyte")) {
ccprintf("Setting 0x%02x to 0x%02x\n", offset, data);
rv = raw_write8(offset, data);
} else
return EC_ERROR_PARAM1;
return rv;
}
DECLARE_CONSOLE_COMMAND(g78x, command_g78x,
"[settemp|setbyte <offset> <value>] or [getbyte <offset>]. "
"Temps in Celsius.",
"Print g781/g782 temp sensor status or set parameters.", NULL);
#endif