summaryrefslogtreecommitdiff
path: root/driver/temp_sensor
diff options
context:
space:
mode:
authorRandall Spangler <rspangler@chromium.org>2013-10-16 13:23:10 -0700
committerchrome-internal-fetch <chrome-internal-fetch@google.com>2013-10-23 20:07:25 +0000
commit8cf03ac0563294fbdeca2dc133d06f0b51c9a546 (patch)
tree6b07c493e7567a3221d8592b4337d2787d6bc531 /driver/temp_sensor
parent2464d08e4d310a3f63208f22df4502c5250c4b58 (diff)
downloadchrome-ec-8cf03ac0563294fbdeca2dc133d06f0b51c9a546.tar.gz
Move source files to driver/ and power/ subdirs
The common/ subdir was getting cluttered. Move drivers for external components to a new driver/ tree, and move what used to be called chipset_*.c to a new power/ directory. This does not move/rename header files or CONFIG options. That will be done in subsequent steps, since moving and modifying .c files in the same CL is harder to review. BUG=chrome-os-partner:18343 BRANCH=none TEST=build all boards; pass unit tests Change-Id: I67a3003dc8564783a320335cf0e9620a21982d5e Signed-off-by: Randall Spangler <rspangler@chromium.org> Reviewed-on: https://chromium-review.googlesource.com/173601 Reviewed-by: Bill Richardson <wfrichar@chromium.org> Tested-by: Bill Richardson <wfrichar@chromium.org> Reviewed-by: Vic Yang <victoryang@chromium.org>
Diffstat (limited to 'driver/temp_sensor')
-rw-r--r--driver/temp_sensor/g781.c210
-rw-r--r--driver/temp_sensor/tmp006.c444
2 files changed, 654 insertions, 0 deletions
diff --git a/driver/temp_sensor/g781.c b/driver/temp_sensor/g781.c
new file mode 100644
index 0000000000..389b12629e
--- /dev/null
+++ b/driver/temp_sensor/g781.c
@@ -0,0 +1,210 @@
+/* Copyright (c) 2013 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 temperature sensor module for Chrome EC */
+
+#include "common.h"
+#include "console.h"
+#include "gpio.h"
+#include "i2c.h"
+#include "hooks.h"
+#include "temp_sensor_g781.h"
+#include "util.h"
+
+static int g781_temp_val_local;
+static int g781_temp_val_remote;
+
+/**
+ * Determine whether the sensor is powered.
+ *
+ * @return non-zero the g781 sensor is powered.
+ */
+static int g781_has_power(void)
+{
+#ifdef CONFIG_TEMP_SENSOR_POWER_GPIO
+ return gpio_get_level(CONFIG_TEMP_SENSOR_POWER_GPIO);
+#else
+ return 1;
+#endif
+}
+
+static int g781_read8(const int offset, int *data_ptr)
+{
+ return i2c_read8(I2C_PORT_THERMAL, G781_I2C_ADDR, offset, data_ptr);
+}
+
+static int g781_write8(const int offset, int data)
+{
+ return i2c_write8(I2C_PORT_THERMAL, G781_I2C_ADDR, offset, data);
+}
+
+static int g781_get_temp(const int offset, int *temp_ptr)
+{
+ int rv;
+ int temp_raw = 0;
+
+ rv = g781_read8(offset, &temp_raw);
+ if (rv < 0)
+ return rv;
+
+ *temp_ptr = (int)(int8_t)temp_raw;
+ return EC_SUCCESS;
+}
+
+static int g781_set_temp(const int offset, int temp)
+{
+ if (temp < -127 || temp > 127)
+ return EC_ERROR_INVAL;
+
+ return g781_write8(offset, (uint8_t)temp);
+}
+
+int g781_get_val(int idx, int *temp_ptr)
+{
+ if (!g781_has_power())
+ return EC_ERROR_NOT_POWERED;
+
+ switch (idx) {
+ case G781_IDX_INTERNAL:
+ *temp_ptr = g781_temp_val_local;
+ break;
+ case G781_IDX_EXTERNAL:
+ *temp_ptr = g781_temp_val_remote;
+ break;
+ default:
+ return EC_ERROR_UNKNOWN;
+ }
+
+ return EC_SUCCESS;
+}
+
+static void g781_temp_sensor_poll(void)
+{
+ if (!g781_has_power())
+ return;
+
+ g781_get_temp(G781_TEMP_LOCAL, &g781_temp_val_local);
+ g781_temp_val_local = C_TO_K(g781_temp_val_local);
+
+ g781_get_temp(G781_TEMP_REMOTE, &g781_temp_val_remote);
+ g781_temp_val_remote = C_TO_K(g781_temp_val_remote);
+}
+DECLARE_HOOK(HOOK_SECOND, g781_temp_sensor_poll, HOOK_PRIO_TEMP_SENSOR);
+
+static int g781_show_status(void)
+{
+ int value;
+ int rv;
+
+
+ rv = g781_get_temp(G781_TEMP_LOCAL, &value);
+ if (rv < 0)
+ return rv;
+ ccprintf("Local Temp: %3dC\n", value);
+
+ rv = g781_get_temp(G781_LOCAL_TEMP_THERM_LIMIT, &value);
+ if (rv < 0)
+ return rv;
+ ccprintf(" Therm Trip: %3dC\n", value);
+
+ rv = g781_get_temp(G781_LOCAL_TEMP_HIGH_LIMIT_R, &value);
+ if (rv < 0)
+ return rv;
+ ccprintf(" High Alarm: %3dC\n", value);
+
+ rv = g781_get_temp(G781_LOCAL_TEMP_LOW_LIMIT_R, &value);
+ if (rv < 0)
+ return rv;
+ ccprintf(" Low Alarm: %3dC\n", value);
+
+ rv = g781_get_temp(G781_TEMP_REMOTE, &value);
+ if (rv < 0)
+ return rv;
+ ccprintf("Remote Temp: %3dC\n", value);
+
+ rv = g781_get_temp(G781_REMOTE_TEMP_THERM_LIMIT, &value);
+ if (rv < 0)
+ return rv;
+ ccprintf(" Therm Trip: %3dC\n", value);
+
+ rv = g781_get_temp(G781_REMOTE_TEMP_HIGH_LIMIT_R, &value);
+ if (rv < 0)
+ return rv;
+ ccprintf(" High Alarm: %3dC\n", value);
+
+ rv = g781_get_temp(G781_REMOTE_TEMP_LOW_LIMIT_R, &value);
+ if (rv < 0)
+ return rv;
+ ccprintf(" Low Alarm: %3dC\n", value);
+
+ rv = g781_read8(G781_STATUS, &value);
+ if (rv < 0)
+ return rv;
+ ccprintf("\nSTATUS: %08b\n", value);
+
+ rv = g781_read8(G781_CONFIGURATION_R, &value);
+ if (rv < 0)
+ return rv;
+ ccprintf("CONFIG: %08b\n", value);
+
+ return EC_SUCCESS;
+}
+
+static int command_g781(int argc, char **argv)
+{
+ char *command;
+ char *e;
+ int data;
+ int offset;
+ int rv;
+
+ if (!g781_has_power()) {
+ ccprintf("ERROR: Temp sensor not powered.\n");
+ return EC_ERROR_NOT_POWERED;
+ }
+
+ /* If no args just print status */
+ if (argc == 1)
+ return g781_show_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 = g781_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 "g781 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 = g781_set_temp(offset, data);
+ } else if (!strcasecmp(command, "setbyte")) {
+ ccprintf("Setting 0x%02x to 0x%02x\n", offset, data);
+ rv = g781_write8(offset, data);
+ } else
+ return EC_ERROR_PARAM1;
+
+ return rv;
+}
+DECLARE_CONSOLE_COMMAND(g781, command_g781,
+ "[settemp|setbyte <offset> <value>] or [getbyte <offset>]. "
+ "Temps in Celsius.",
+ "Print g781 temp sensor status or set parameters.", NULL);
diff --git a/driver/temp_sensor/tmp006.c b/driver/temp_sensor/tmp006.c
new file mode 100644
index 0000000000..19d66d87aa
--- /dev/null
+++ b/driver/temp_sensor/tmp006.c
@@ -0,0 +1,444 @@
+/* Copyright (c) 2012 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.
+ */
+
+/* TMP006 temperature sensor module for Chrome EC */
+
+#include "common.h"
+#include "console.h"
+#include "gpio.h"
+#include "hooks.h"
+#include "host_command.h"
+#include "i2c.h"
+#include "math.h"
+#include "task.h"
+#include "temp_sensor.h"
+#include "tmp006.h"
+#include "util.h"
+
+/* Console output macros */
+#define CPUTS(outstr) cputs(CC_THERMAL, outstr)
+#define CPRINTF(format, args...) cprintf(CC_THERMAL, format, ## args)
+
+/* Constants for calculating target object temperatures */
+static const float A1 = 1.75e-3f;
+static const float A2 = -1.678e-5f;
+static const float B0 = -2.94e-5f;
+static const float B1 = -5.7e-7f;
+static const float B2 = 4.63e-9f;
+static const float C2 = 13.4f;
+
+/* Defined in board_temp_sensor.c. */
+extern const struct tmp006_t tmp006_sensors[TMP006_COUNT];
+
+/* Flags for tdata->fail */
+#define FAIL_INIT (1 << 0) /* Just initialized */
+#define FAIL_POWER (1 << 1) /* Sensor not powered */
+#define FAIL_I2C (1 << 2) /* I2C communication error */
+#define FAIL_NOT_READY (1 << 3) /* Data not ready */
+
+struct tmp006_data_t {
+ int v; /* Object voltage */
+ int t[4]; /* Circular buffer of last four die temperatures */
+ int tidx; /* Index of the current value in t[] */
+ int fail; /* Fail flags; non-zero if last read failed */
+ float s0; /* Sensitivity factor */
+ float b0, b1, b2; /* Coefficients for self-heating correction */
+};
+
+static struct tmp006_data_t tmp006_data[TMP006_COUNT];
+
+/**
+ * Check if sensor has power
+ *
+ * @param idx Sensor index
+ *
+ * @return non-zero if sensor has power.
+ */
+static int tmp006_has_power(int idx)
+{
+#ifdef CONFIG_TEMP_SENSOR_POWER_GPIO
+ return gpio_get_level(CONFIG_TEMP_SENSOR_POWER_GPIO);
+#else
+ return 1;
+#endif
+}
+
+static int tmp006_read_die_temp(const struct tmp006_data_t *tdata,
+ int *temp_ptr)
+{
+ if (tdata->fail)
+ return EC_ERROR_UNKNOWN;
+
+ /* Return previous die temperature */
+ *temp_ptr = tdata->t[(tdata->tidx - 1) & 0x3] / 100;
+ return EC_SUCCESS;
+}
+
+/**
+ * Calculate the remote object temperature.
+ *
+ * @param Tdie_i Die temperature in 1/100 K.
+ * @param Vobj_i Voltage read from register 0. In nV.
+ * @param tdata TMP006 data for this sensor.
+ *
+ * @return Object temperature in 1/100 K.
+ */
+static int tmp006_calculate_object_temp(int Tdie_i, int Vobj_i,
+ const struct tmp006_data_t *tdata)
+{
+ float Tdie, Vobj;
+ float Tx, S, Vos, Vx, fv, Tobj, T4;
+ int Tobj_i;
+
+ Tdie = (float)Tdie_i * 1e-2f;
+ Vobj = (float)Vobj_i * 1e-9f;
+
+ /* Calculate according to TMP006 users guide. */
+ Tx = Tdie - 298.15f;
+ /* S is the sensitivity */
+ S = tdata->s0 * (1.0f + A1 * Tx + A2 * Tx * Tx);
+ /* Vos is the offset voltage */
+ Vos = tdata->b0 + tdata->b1 * Tx + tdata->b2 * Tx * Tx;
+ Vx = Vobj - Vos;
+ /* fv is Seebeck coefficient f(Vobj) */
+ fv = Vx + C2 * Vx * Vx;
+
+ T4 = Tdie * Tdie * Tdie * Tdie + fv / S;
+ Tobj = sqrtf(sqrtf(T4));
+ Tobj_i = (int32_t)(Tobj * 100.0f);
+
+ return Tobj_i;
+}
+
+/**
+ * Apply TMP006 temporal correction.
+ *
+ * @param T1-T4 Four die temperature readings separated by 1s in 1/100K.
+ * @param Vobj Voltage read from register 0, in nV.
+ *
+ * @return Corrected object voltage in 1/100K.
+ */
+static int tmp006_correct_object_voltage(int T1, int T2, int T3, int T4,
+ int Vobj)
+{
+ int Tslope = 3 * T1 + T2 - T3 - 3 * T4;
+ return Vobj + 296 * Tslope;
+}
+
+static int tmp006_read_object_temp(const struct tmp006_data_t *tdata,
+ int *temp_ptr)
+{
+ int pidx = (tdata->tidx - 1) & 0x3;
+ int t = tdata->t[pidx];
+ int v = tdata->v;
+
+ if (tdata->fail)
+ return EC_ERROR_UNKNOWN;
+
+ if (!tdata->s0)
+ return EC_ERROR_NOT_CALIBRATED;
+
+ v = tmp006_correct_object_voltage(
+ t,
+ tdata->t[(pidx + 3) & 3],
+ tdata->t[(pidx + 2) & 3],
+ tdata->t[(pidx + 1) & 3],
+ v);
+
+ *temp_ptr = tmp006_calculate_object_temp(t, v, tdata) / 100;
+
+ return EC_SUCCESS;
+}
+
+static int tmp006_poll_sensor(int sensor_id)
+{
+ struct tmp006_data_t *tdata = tmp006_data + sensor_id;
+ int traw, t;
+ int vraw, v;
+ int rv;
+ int addr = tmp006_sensors[sensor_id].addr;
+ int idx;
+
+ if (!tmp006_has_power(sensor_id)) {
+ tdata->fail |= FAIL_POWER;
+ return EC_ERROR_UNKNOWN;
+ }
+
+ /*
+ * If sensor has just initialized and/or has lost power, wait for
+ * data ready; otherwise, we read garbage data.
+ */
+ if (tdata->fail && (FAIL_POWER | FAIL_INIT)) {
+ rv = i2c_read16(TMP006_PORT(addr), TMP006_REG(addr), 0x02, &v);
+ if (rv) {
+ tdata->fail |= FAIL_I2C;
+ return EC_ERROR_UNKNOWN;
+ } else if (!(v & 0x80)) {
+ tdata->fail |= FAIL_NOT_READY;
+ return EC_ERROR_UNKNOWN;
+ }
+ }
+
+ rv = i2c_read16(TMP006_PORT(addr), TMP006_REG(addr), 0x01, &traw);
+ if (rv) {
+ tdata->fail |= FAIL_I2C;
+ return EC_ERROR_UNKNOWN;
+ }
+
+ /* Convert temperature from raw to 1/100 K */
+ t = ((int)(int16_t)traw * 100) / 128 + 27300;
+
+ rv = i2c_read16(TMP006_PORT(addr), TMP006_REG(addr), 0x00, &vraw);
+ if (rv) {
+ tdata->fail |= FAIL_I2C;
+ return EC_ERROR_UNKNOWN;
+ }
+
+ /* Convert voltage from raw to nV */
+ v = ((int)(int16_t)vraw * 15625) / 100;
+
+ /*
+ * If last read failed, set the entire temperature history to the
+ * current temperature. This keeps us from making inaccurate temporal
+ * corrections based on stale data.
+ */
+ if (tdata->fail) {
+ for (idx = 0; idx < 4; idx++)
+ tdata->t[idx] = t;
+ } else {
+ idx = tdata->tidx;
+ tdata->t[idx] = t;
+ tdata->tidx = (idx + 1) & 3;
+ }
+
+ tdata->v = v;
+ tdata->fail = 0;
+
+ return EC_SUCCESS;
+}
+
+int tmp006_get_val(int idx, int *temp_ptr)
+{
+ /*
+ * Note: idx is a thermal sensor index, where the top N-1 bits are the
+ * TMP006 index and the bottom bit is (0=die, 1=remote).
+ */
+ int tidx = idx >> 1;
+ const struct tmp006_data_t *tdata = tmp006_data + tidx;
+
+ if (tdata->fail & FAIL_POWER) {
+ /*
+ * Sensor isn't powered, or hasn't successfully provided data
+ * since being powered. Keep reporting not-powered until
+ * we get good data (which will clear FAIL_POWER) or there is
+ * an I2C error.
+ */
+ return (tdata->fail & FAIL_I2C) ? EC_ERROR_UNKNOWN :
+ EC_ERROR_NOT_POWERED;
+ }
+
+ /* Check the low bit to determine which temperature to read. */
+ if ((idx & 0x1) == 0)
+ return tmp006_read_die_temp(tdata, temp_ptr);
+ else
+ return tmp006_read_object_temp(tdata, temp_ptr);
+}
+
+/*****************************************************************************/
+/* Hooks */
+
+static void tmp006_poll(void)
+{
+ int i;
+
+ for (i = 0; i < TMP006_COUNT; ++i)
+ tmp006_poll_sensor(i);
+}
+DECLARE_HOOK(HOOK_SECOND, tmp006_poll, HOOK_PRIO_TEMP_SENSOR);
+
+static void tmp006_init(void)
+{
+ int i;
+
+ for (i = 0; i < TMP006_COUNT; ++i) {
+ struct tmp006_data_t *tdata = tmp006_data + i;
+
+ /* Report error until we actually read the sensor */
+ tdata->fail = FAIL_INIT;
+
+ /* Use defaults for Bn params */
+ tdata->b0 = B0;
+ tdata->b1 = B1;
+ tdata->b2 = B2;
+ }
+}
+DECLARE_HOOK(HOOK_INIT, tmp006_init, HOOK_PRIO_DEFAULT);
+
+/*****************************************************************************/
+/* Host commands */
+
+int tmp006_get_calibration(struct host_cmd_handler_args *args)
+{
+ const struct ec_params_tmp006_get_calibration *p = args->params;
+ struct ec_response_tmp006_get_calibration *r = args->response;
+ const struct tmp006_data_t *tdata;
+
+ if (p->index >= TMP006_COUNT)
+ return EC_RES_INVALID_PARAM;
+
+ tdata = tmp006_data + p->index;
+
+ r->s0 = tdata->s0;
+ r->b0 = tdata->b0;
+ r->b1 = tdata->b1;
+ r->b2 = tdata->b2;
+
+ args->response_size = sizeof(*r);
+
+ return EC_RES_SUCCESS;
+}
+DECLARE_HOST_COMMAND(EC_CMD_TMP006_GET_CALIBRATION,
+ tmp006_get_calibration,
+ EC_VER_MASK(0));
+
+int tmp006_set_calibration(struct host_cmd_handler_args *args)
+{
+ const struct ec_params_tmp006_set_calibration *p = args->params;
+ struct tmp006_data_t *tdata;
+
+ if (p->index >= TMP006_COUNT)
+ return EC_RES_INVALID_PARAM;
+
+ tdata = tmp006_data + p->index;
+
+ tdata->s0 = p->s0;
+ tdata->b0 = p->b0;
+ tdata->b1 = p->b1;
+ tdata->b2 = p->b2;
+
+ return EC_RES_SUCCESS;
+}
+DECLARE_HOST_COMMAND(EC_CMD_TMP006_SET_CALIBRATION,
+ tmp006_set_calibration,
+ EC_VER_MASK(0));
+
+/*****************************************************************************/
+/* Console commands */
+
+/**
+ * Print temperature info for a sensor; used by console command.
+ */
+static int tmp006_print(int idx)
+{
+ int vraw, v;
+ int traw, t;
+ int rv;
+ int d;
+ int addr = tmp006_sensors[idx].addr;
+
+
+ ccprintf("Debug data from %s:\n", tmp006_sensors[idx].name);
+
+ if (!tmp006_has_power(idx)) {
+ ccputs("Sensor powered off.\n");
+ return EC_ERROR_UNKNOWN;
+ }
+
+ rv = i2c_read16(TMP006_PORT(addr), TMP006_REG(addr), 0xfe, &d);
+ if (rv)
+ return rv;
+ ccprintf(" Manufacturer ID: 0x%04x\n", d);
+
+ rv = i2c_read16(TMP006_PORT(addr), TMP006_REG(addr), 0xff, &d);
+ ccprintf(" Device ID: 0x%04x\n", d);
+
+ rv = i2c_read16(TMP006_PORT(addr), TMP006_REG(addr), 0x02, &d);
+ ccprintf(" Config: 0x%04x\n", d);
+
+ rv = i2c_read16(TMP006_PORT(addr), TMP006_REG(addr), 0x00, &vraw);
+ v = ((int)(int16_t)vraw * 15625) / 100;
+ ccprintf(" Voltage: 0x%04x = %d nV\n", vraw, v);
+
+ rv = i2c_read16(TMP006_PORT(addr), TMP006_REG(addr), 0x01, &traw);
+ t = ((int)(int16_t)traw * 100) / 128;
+ ccprintf(" Temperature: 0x%04x = %d.%02d C\n",
+ traw, t / 100, t > 0 ? t % 100 : 100 - (t % 100));
+
+ return EC_SUCCESS;
+}
+
+static int command_sensor_info(int argc, char **argv)
+{
+ int i;
+ int rv, rv1;
+
+ rv1 = EC_SUCCESS;
+ for (i = 0; i < TMP006_COUNT; i++) {
+ rv = tmp006_print(i);
+ if (rv != EC_SUCCESS)
+ rv1 = rv;
+ cflush();
+ }
+
+ return rv1;
+}
+DECLARE_CONSOLE_COMMAND(tmp006, command_sensor_info,
+ NULL,
+ "Print TMP006 sensors",
+ NULL);
+
+static int command_t6cal(int argc, char **argv)
+{
+ struct tmp006_data_t *tdata;
+ char *e;
+ int v;
+ int i;
+
+ if (argc < 2) {
+ ccprintf("# Name S0 b0"
+ " b1 b2\n");
+ for (i = 0; i < TMP006_COUNT; i++) {
+ tdata = tmp006_data + i;
+ ccprintf("%d %-11s"
+ "%7de-17 %7de-8 %7de-10 %7de-12\n",
+ i, tmp006_sensors[i].name,
+ (int)(tdata->s0 * 1e17f),
+ (int)(tdata->b0 * 1e8f),
+ (int)(tdata->b1 * 1e10f),
+ (int)(tdata->b2 * 1e12f));
+ }
+
+ return EC_SUCCESS;
+ }
+
+ if (argc != 4)
+ return EC_ERROR_PARAM_COUNT;
+
+ i = strtoi(argv[1], &e, 0);
+ if (*e || i < 0 || i >= TMP006_COUNT)
+ return EC_ERROR_PARAM1;
+ tdata = tmp006_data + i;
+
+ v = strtoi(argv[3], &e, 0);
+ if (*e)
+ return EC_ERROR_PARAM3;
+
+ if (!strcasecmp(argv[2], "s0"))
+ tdata->s0 = (float)v * 1e-17f;
+ else if (!strcasecmp(argv[2], "b0"))
+ tdata->b0 = (float)v * 1e-8f;
+ else if (!strcasecmp(argv[2], "b1"))
+ tdata->b1 = (float)v * 1e-10f;
+ else if (!strcasecmp(argv[2], "b2"))
+ tdata->b2 = (float)v * 1e-12f;
+ else
+ return EC_ERROR_PARAM2;
+
+ return EC_SUCCESS;
+}
+DECLARE_CONSOLE_COMMAND(t6cal, command_t6cal,
+ "[<index> <coeff_name> <radix>]",
+ "Set/print TMP006 calibration",
+ NULL);