/* 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_FLAGS, 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_FLAGS, 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: %pb\n", BINARY_VALUE(value, 8)); #ifdef CONFIG_TEMP_SENSOR_G782 if (raw_read8(G78X_STATUS1, &value) == EC_SUCCESS) ccprintf("STATUS1: %pb\n", BINARY_VALUE(value, 8)); #endif if (raw_read8(G78X_CONFIGURATION_R, &value) == EC_SUCCESS) ccprintf("CONFIG: %pb\n", BINARY_VALUE(value, 8)); 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 %pb\n", offset, BINARY_VALUE(data, 8)); 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 ] or [getbyte ]. " "Temps in Celsius.", "Print g781/g782 temp sensor status or set parameters."); #endif