1*5e9ae4b1SCorey Minyard /* 2*5e9ae4b1SCorey Minyard * Texas Instruments TMP105 temperature sensor. 3*5e9ae4b1SCorey Minyard * 4*5e9ae4b1SCorey Minyard * Copyright (C) 2008 Nokia Corporation 5*5e9ae4b1SCorey Minyard * Written by Andrzej Zaborowski <andrew@openedhand.com> 6*5e9ae4b1SCorey Minyard * 7*5e9ae4b1SCorey Minyard * This program is free software; you can redistribute it and/or 8*5e9ae4b1SCorey Minyard * modify it under the terms of the GNU General Public License as 9*5e9ae4b1SCorey Minyard * published by the Free Software Foundation; either version 2 or 10*5e9ae4b1SCorey Minyard * (at your option) version 3 of the License. 11*5e9ae4b1SCorey Minyard * 12*5e9ae4b1SCorey Minyard * This program is distributed in the hope that it will be useful, 13*5e9ae4b1SCorey Minyard * but WITHOUT ANY WARRANTY; without even the implied warranty of 14*5e9ae4b1SCorey Minyard * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 15*5e9ae4b1SCorey Minyard * GNU General Public License for more details. 16*5e9ae4b1SCorey Minyard * 17*5e9ae4b1SCorey Minyard * You should have received a copy of the GNU General Public License along 18*5e9ae4b1SCorey Minyard * with this program; if not, see <http://www.gnu.org/licenses/>. 19*5e9ae4b1SCorey Minyard */ 20*5e9ae4b1SCorey Minyard 21*5e9ae4b1SCorey Minyard #include "qemu/osdep.h" 22*5e9ae4b1SCorey Minyard #include "hw/i2c/i2c.h" 23*5e9ae4b1SCorey Minyard #include "hw/irq.h" 24*5e9ae4b1SCorey Minyard #include "migration/vmstate.h" 25*5e9ae4b1SCorey Minyard #include "hw/sensor/tmp105.h" 26*5e9ae4b1SCorey Minyard #include "qapi/error.h" 27*5e9ae4b1SCorey Minyard #include "qapi/visitor.h" 28*5e9ae4b1SCorey Minyard #include "qemu/module.h" 29*5e9ae4b1SCorey Minyard 30*5e9ae4b1SCorey Minyard static void tmp105_interrupt_update(TMP105State *s) 31*5e9ae4b1SCorey Minyard { 32*5e9ae4b1SCorey Minyard qemu_set_irq(s->pin, s->alarm ^ ((~s->config >> 2) & 1)); /* POL */ 33*5e9ae4b1SCorey Minyard } 34*5e9ae4b1SCorey Minyard 35*5e9ae4b1SCorey Minyard static void tmp105_alarm_update(TMP105State *s) 36*5e9ae4b1SCorey Minyard { 37*5e9ae4b1SCorey Minyard if ((s->config >> 0) & 1) { /* SD */ 38*5e9ae4b1SCorey Minyard if ((s->config >> 7) & 1) /* OS */ 39*5e9ae4b1SCorey Minyard s->config &= ~(1 << 7); /* OS */ 40*5e9ae4b1SCorey Minyard else 41*5e9ae4b1SCorey Minyard return; 42*5e9ae4b1SCorey Minyard } 43*5e9ae4b1SCorey Minyard 44*5e9ae4b1SCorey Minyard if (s->config >> 1 & 1) { 45*5e9ae4b1SCorey Minyard /* 46*5e9ae4b1SCorey Minyard * TM == 1 : Interrupt mode. We signal Alert when the 47*5e9ae4b1SCorey Minyard * temperature rises above T_high, and expect the guest to clear 48*5e9ae4b1SCorey Minyard * it (eg by reading a device register). 49*5e9ae4b1SCorey Minyard */ 50*5e9ae4b1SCorey Minyard if (s->detect_falling) { 51*5e9ae4b1SCorey Minyard if (s->temperature < s->limit[0]) { 52*5e9ae4b1SCorey Minyard s->alarm = 1; 53*5e9ae4b1SCorey Minyard s->detect_falling = false; 54*5e9ae4b1SCorey Minyard } 55*5e9ae4b1SCorey Minyard } else { 56*5e9ae4b1SCorey Minyard if (s->temperature >= s->limit[1]) { 57*5e9ae4b1SCorey Minyard s->alarm = 1; 58*5e9ae4b1SCorey Minyard s->detect_falling = true; 59*5e9ae4b1SCorey Minyard } 60*5e9ae4b1SCorey Minyard } 61*5e9ae4b1SCorey Minyard } else { 62*5e9ae4b1SCorey Minyard /* 63*5e9ae4b1SCorey Minyard * TM == 0 : Comparator mode. We signal Alert when the temperature 64*5e9ae4b1SCorey Minyard * rises above T_high, and stop signalling it when the temperature 65*5e9ae4b1SCorey Minyard * falls below T_low. 66*5e9ae4b1SCorey Minyard */ 67*5e9ae4b1SCorey Minyard if (s->detect_falling) { 68*5e9ae4b1SCorey Minyard if (s->temperature < s->limit[0]) { 69*5e9ae4b1SCorey Minyard s->alarm = 0; 70*5e9ae4b1SCorey Minyard s->detect_falling = false; 71*5e9ae4b1SCorey Minyard } 72*5e9ae4b1SCorey Minyard } else { 73*5e9ae4b1SCorey Minyard if (s->temperature >= s->limit[1]) { 74*5e9ae4b1SCorey Minyard s->alarm = 1; 75*5e9ae4b1SCorey Minyard s->detect_falling = true; 76*5e9ae4b1SCorey Minyard } 77*5e9ae4b1SCorey Minyard } 78*5e9ae4b1SCorey Minyard } 79*5e9ae4b1SCorey Minyard 80*5e9ae4b1SCorey Minyard tmp105_interrupt_update(s); 81*5e9ae4b1SCorey Minyard } 82*5e9ae4b1SCorey Minyard 83*5e9ae4b1SCorey Minyard static void tmp105_get_temperature(Object *obj, Visitor *v, const char *name, 84*5e9ae4b1SCorey Minyard void *opaque, Error **errp) 85*5e9ae4b1SCorey Minyard { 86*5e9ae4b1SCorey Minyard TMP105State *s = TMP105(obj); 87*5e9ae4b1SCorey Minyard int64_t value = s->temperature * 1000 / 256; 88*5e9ae4b1SCorey Minyard 89*5e9ae4b1SCorey Minyard visit_type_int(v, name, &value, errp); 90*5e9ae4b1SCorey Minyard } 91*5e9ae4b1SCorey Minyard 92*5e9ae4b1SCorey Minyard /* Units are 0.001 centigrades relative to 0 C. s->temperature is 8.8 93*5e9ae4b1SCorey Minyard * fixed point, so units are 1/256 centigrades. A simple ratio will do. 94*5e9ae4b1SCorey Minyard */ 95*5e9ae4b1SCorey Minyard static void tmp105_set_temperature(Object *obj, Visitor *v, const char *name, 96*5e9ae4b1SCorey Minyard void *opaque, Error **errp) 97*5e9ae4b1SCorey Minyard { 98*5e9ae4b1SCorey Minyard TMP105State *s = TMP105(obj); 99*5e9ae4b1SCorey Minyard int64_t temp; 100*5e9ae4b1SCorey Minyard 101*5e9ae4b1SCorey Minyard if (!visit_type_int(v, name, &temp, errp)) { 102*5e9ae4b1SCorey Minyard return; 103*5e9ae4b1SCorey Minyard } 104*5e9ae4b1SCorey Minyard if (temp >= 128000 || temp < -128000) { 105*5e9ae4b1SCorey Minyard error_setg(errp, "value %" PRId64 ".%03" PRIu64 " C is out of range", 106*5e9ae4b1SCorey Minyard temp / 1000, temp % 1000); 107*5e9ae4b1SCorey Minyard return; 108*5e9ae4b1SCorey Minyard } 109*5e9ae4b1SCorey Minyard 110*5e9ae4b1SCorey Minyard s->temperature = (int16_t) (temp * 256 / 1000); 111*5e9ae4b1SCorey Minyard 112*5e9ae4b1SCorey Minyard tmp105_alarm_update(s); 113*5e9ae4b1SCorey Minyard } 114*5e9ae4b1SCorey Minyard 115*5e9ae4b1SCorey Minyard static const int tmp105_faultq[4] = { 1, 2, 4, 6 }; 116*5e9ae4b1SCorey Minyard 117*5e9ae4b1SCorey Minyard static void tmp105_read(TMP105State *s) 118*5e9ae4b1SCorey Minyard { 119*5e9ae4b1SCorey Minyard s->len = 0; 120*5e9ae4b1SCorey Minyard 121*5e9ae4b1SCorey Minyard if ((s->config >> 1) & 1) { /* TM */ 122*5e9ae4b1SCorey Minyard s->alarm = 0; 123*5e9ae4b1SCorey Minyard tmp105_interrupt_update(s); 124*5e9ae4b1SCorey Minyard } 125*5e9ae4b1SCorey Minyard 126*5e9ae4b1SCorey Minyard switch (s->pointer & 3) { 127*5e9ae4b1SCorey Minyard case TMP105_REG_TEMPERATURE: 128*5e9ae4b1SCorey Minyard s->buf[s->len ++] = (((uint16_t) s->temperature) >> 8); 129*5e9ae4b1SCorey Minyard s->buf[s->len ++] = (((uint16_t) s->temperature) >> 0) & 130*5e9ae4b1SCorey Minyard (0xf0 << ((~s->config >> 5) & 3)); /* R */ 131*5e9ae4b1SCorey Minyard break; 132*5e9ae4b1SCorey Minyard 133*5e9ae4b1SCorey Minyard case TMP105_REG_CONFIG: 134*5e9ae4b1SCorey Minyard s->buf[s->len ++] = s->config; 135*5e9ae4b1SCorey Minyard break; 136*5e9ae4b1SCorey Minyard 137*5e9ae4b1SCorey Minyard case TMP105_REG_T_LOW: 138*5e9ae4b1SCorey Minyard s->buf[s->len ++] = ((uint16_t) s->limit[0]) >> 8; 139*5e9ae4b1SCorey Minyard s->buf[s->len ++] = ((uint16_t) s->limit[0]) >> 0; 140*5e9ae4b1SCorey Minyard break; 141*5e9ae4b1SCorey Minyard 142*5e9ae4b1SCorey Minyard case TMP105_REG_T_HIGH: 143*5e9ae4b1SCorey Minyard s->buf[s->len ++] = ((uint16_t) s->limit[1]) >> 8; 144*5e9ae4b1SCorey Minyard s->buf[s->len ++] = ((uint16_t) s->limit[1]) >> 0; 145*5e9ae4b1SCorey Minyard break; 146*5e9ae4b1SCorey Minyard } 147*5e9ae4b1SCorey Minyard } 148*5e9ae4b1SCorey Minyard 149*5e9ae4b1SCorey Minyard static void tmp105_write(TMP105State *s) 150*5e9ae4b1SCorey Minyard { 151*5e9ae4b1SCorey Minyard switch (s->pointer & 3) { 152*5e9ae4b1SCorey Minyard case TMP105_REG_TEMPERATURE: 153*5e9ae4b1SCorey Minyard break; 154*5e9ae4b1SCorey Minyard 155*5e9ae4b1SCorey Minyard case TMP105_REG_CONFIG: 156*5e9ae4b1SCorey Minyard if (s->buf[0] & ~s->config & (1 << 0)) /* SD */ 157*5e9ae4b1SCorey Minyard printf("%s: TMP105 shutdown\n", __func__); 158*5e9ae4b1SCorey Minyard s->config = s->buf[0]; 159*5e9ae4b1SCorey Minyard s->faults = tmp105_faultq[(s->config >> 3) & 3]; /* F */ 160*5e9ae4b1SCorey Minyard tmp105_alarm_update(s); 161*5e9ae4b1SCorey Minyard break; 162*5e9ae4b1SCorey Minyard 163*5e9ae4b1SCorey Minyard case TMP105_REG_T_LOW: 164*5e9ae4b1SCorey Minyard case TMP105_REG_T_HIGH: 165*5e9ae4b1SCorey Minyard if (s->len >= 3) 166*5e9ae4b1SCorey Minyard s->limit[s->pointer & 1] = (int16_t) 167*5e9ae4b1SCorey Minyard ((((uint16_t) s->buf[0]) << 8) | s->buf[1]); 168*5e9ae4b1SCorey Minyard tmp105_alarm_update(s); 169*5e9ae4b1SCorey Minyard break; 170*5e9ae4b1SCorey Minyard } 171*5e9ae4b1SCorey Minyard } 172*5e9ae4b1SCorey Minyard 173*5e9ae4b1SCorey Minyard static uint8_t tmp105_rx(I2CSlave *i2c) 174*5e9ae4b1SCorey Minyard { 175*5e9ae4b1SCorey Minyard TMP105State *s = TMP105(i2c); 176*5e9ae4b1SCorey Minyard 177*5e9ae4b1SCorey Minyard if (s->len < 2) { 178*5e9ae4b1SCorey Minyard return s->buf[s->len ++]; 179*5e9ae4b1SCorey Minyard } else { 180*5e9ae4b1SCorey Minyard return 0xff; 181*5e9ae4b1SCorey Minyard } 182*5e9ae4b1SCorey Minyard } 183*5e9ae4b1SCorey Minyard 184*5e9ae4b1SCorey Minyard static int tmp105_tx(I2CSlave *i2c, uint8_t data) 185*5e9ae4b1SCorey Minyard { 186*5e9ae4b1SCorey Minyard TMP105State *s = TMP105(i2c); 187*5e9ae4b1SCorey Minyard 188*5e9ae4b1SCorey Minyard if (s->len == 0) { 189*5e9ae4b1SCorey Minyard s->pointer = data; 190*5e9ae4b1SCorey Minyard s->len++; 191*5e9ae4b1SCorey Minyard } else { 192*5e9ae4b1SCorey Minyard if (s->len <= 2) { 193*5e9ae4b1SCorey Minyard s->buf[s->len - 1] = data; 194*5e9ae4b1SCorey Minyard } 195*5e9ae4b1SCorey Minyard s->len++; 196*5e9ae4b1SCorey Minyard tmp105_write(s); 197*5e9ae4b1SCorey Minyard } 198*5e9ae4b1SCorey Minyard 199*5e9ae4b1SCorey Minyard return 0; 200*5e9ae4b1SCorey Minyard } 201*5e9ae4b1SCorey Minyard 202*5e9ae4b1SCorey Minyard static int tmp105_event(I2CSlave *i2c, enum i2c_event event) 203*5e9ae4b1SCorey Minyard { 204*5e9ae4b1SCorey Minyard TMP105State *s = TMP105(i2c); 205*5e9ae4b1SCorey Minyard 206*5e9ae4b1SCorey Minyard if (event == I2C_START_RECV) { 207*5e9ae4b1SCorey Minyard tmp105_read(s); 208*5e9ae4b1SCorey Minyard } 209*5e9ae4b1SCorey Minyard 210*5e9ae4b1SCorey Minyard s->len = 0; 211*5e9ae4b1SCorey Minyard return 0; 212*5e9ae4b1SCorey Minyard } 213*5e9ae4b1SCorey Minyard 214*5e9ae4b1SCorey Minyard static int tmp105_post_load(void *opaque, int version_id) 215*5e9ae4b1SCorey Minyard { 216*5e9ae4b1SCorey Minyard TMP105State *s = opaque; 217*5e9ae4b1SCorey Minyard 218*5e9ae4b1SCorey Minyard s->faults = tmp105_faultq[(s->config >> 3) & 3]; /* F */ 219*5e9ae4b1SCorey Minyard 220*5e9ae4b1SCorey Minyard tmp105_interrupt_update(s); 221*5e9ae4b1SCorey Minyard return 0; 222*5e9ae4b1SCorey Minyard } 223*5e9ae4b1SCorey Minyard 224*5e9ae4b1SCorey Minyard static bool detect_falling_needed(void *opaque) 225*5e9ae4b1SCorey Minyard { 226*5e9ae4b1SCorey Minyard TMP105State *s = opaque; 227*5e9ae4b1SCorey Minyard 228*5e9ae4b1SCorey Minyard /* 229*5e9ae4b1SCorey Minyard * We only need to migrate the detect_falling bool if it's set; 230*5e9ae4b1SCorey Minyard * for migration from older machines we assume that it is false 231*5e9ae4b1SCorey Minyard * (ie temperature is not out of range). 232*5e9ae4b1SCorey Minyard */ 233*5e9ae4b1SCorey Minyard return s->detect_falling; 234*5e9ae4b1SCorey Minyard } 235*5e9ae4b1SCorey Minyard 236*5e9ae4b1SCorey Minyard static const VMStateDescription vmstate_tmp105_detect_falling = { 237*5e9ae4b1SCorey Minyard .name = "TMP105/detect-falling", 238*5e9ae4b1SCorey Minyard .version_id = 1, 239*5e9ae4b1SCorey Minyard .minimum_version_id = 1, 240*5e9ae4b1SCorey Minyard .needed = detect_falling_needed, 241*5e9ae4b1SCorey Minyard .fields = (VMStateField[]) { 242*5e9ae4b1SCorey Minyard VMSTATE_BOOL(detect_falling, TMP105State), 243*5e9ae4b1SCorey Minyard VMSTATE_END_OF_LIST() 244*5e9ae4b1SCorey Minyard } 245*5e9ae4b1SCorey Minyard }; 246*5e9ae4b1SCorey Minyard 247*5e9ae4b1SCorey Minyard static const VMStateDescription vmstate_tmp105 = { 248*5e9ae4b1SCorey Minyard .name = "TMP105", 249*5e9ae4b1SCorey Minyard .version_id = 0, 250*5e9ae4b1SCorey Minyard .minimum_version_id = 0, 251*5e9ae4b1SCorey Minyard .post_load = tmp105_post_load, 252*5e9ae4b1SCorey Minyard .fields = (VMStateField[]) { 253*5e9ae4b1SCorey Minyard VMSTATE_UINT8(len, TMP105State), 254*5e9ae4b1SCorey Minyard VMSTATE_UINT8_ARRAY(buf, TMP105State, 2), 255*5e9ae4b1SCorey Minyard VMSTATE_UINT8(pointer, TMP105State), 256*5e9ae4b1SCorey Minyard VMSTATE_UINT8(config, TMP105State), 257*5e9ae4b1SCorey Minyard VMSTATE_INT16(temperature, TMP105State), 258*5e9ae4b1SCorey Minyard VMSTATE_INT16_ARRAY(limit, TMP105State, 2), 259*5e9ae4b1SCorey Minyard VMSTATE_UINT8(alarm, TMP105State), 260*5e9ae4b1SCorey Minyard VMSTATE_I2C_SLAVE(i2c, TMP105State), 261*5e9ae4b1SCorey Minyard VMSTATE_END_OF_LIST() 262*5e9ae4b1SCorey Minyard }, 263*5e9ae4b1SCorey Minyard .subsections = (const VMStateDescription*[]) { 264*5e9ae4b1SCorey Minyard &vmstate_tmp105_detect_falling, 265*5e9ae4b1SCorey Minyard NULL 266*5e9ae4b1SCorey Minyard } 267*5e9ae4b1SCorey Minyard }; 268*5e9ae4b1SCorey Minyard 269*5e9ae4b1SCorey Minyard static void tmp105_reset(I2CSlave *i2c) 270*5e9ae4b1SCorey Minyard { 271*5e9ae4b1SCorey Minyard TMP105State *s = TMP105(i2c); 272*5e9ae4b1SCorey Minyard 273*5e9ae4b1SCorey Minyard s->temperature = 0; 274*5e9ae4b1SCorey Minyard s->pointer = 0; 275*5e9ae4b1SCorey Minyard s->config = 0; 276*5e9ae4b1SCorey Minyard s->faults = tmp105_faultq[(s->config >> 3) & 3]; 277*5e9ae4b1SCorey Minyard s->alarm = 0; 278*5e9ae4b1SCorey Minyard s->detect_falling = false; 279*5e9ae4b1SCorey Minyard 280*5e9ae4b1SCorey Minyard s->limit[0] = 0x4b00; /* T_LOW, 75 degrees C */ 281*5e9ae4b1SCorey Minyard s->limit[1] = 0x5000; /* T_HIGH, 80 degrees C */ 282*5e9ae4b1SCorey Minyard 283*5e9ae4b1SCorey Minyard tmp105_interrupt_update(s); 284*5e9ae4b1SCorey Minyard } 285*5e9ae4b1SCorey Minyard 286*5e9ae4b1SCorey Minyard static void tmp105_realize(DeviceState *dev, Error **errp) 287*5e9ae4b1SCorey Minyard { 288*5e9ae4b1SCorey Minyard I2CSlave *i2c = I2C_SLAVE(dev); 289*5e9ae4b1SCorey Minyard TMP105State *s = TMP105(i2c); 290*5e9ae4b1SCorey Minyard 291*5e9ae4b1SCorey Minyard qdev_init_gpio_out(&i2c->qdev, &s->pin, 1); 292*5e9ae4b1SCorey Minyard 293*5e9ae4b1SCorey Minyard tmp105_reset(&s->i2c); 294*5e9ae4b1SCorey Minyard } 295*5e9ae4b1SCorey Minyard 296*5e9ae4b1SCorey Minyard static void tmp105_initfn(Object *obj) 297*5e9ae4b1SCorey Minyard { 298*5e9ae4b1SCorey Minyard object_property_add(obj, "temperature", "int", 299*5e9ae4b1SCorey Minyard tmp105_get_temperature, 300*5e9ae4b1SCorey Minyard tmp105_set_temperature, NULL, NULL); 301*5e9ae4b1SCorey Minyard } 302*5e9ae4b1SCorey Minyard 303*5e9ae4b1SCorey Minyard static void tmp105_class_init(ObjectClass *klass, void *data) 304*5e9ae4b1SCorey Minyard { 305*5e9ae4b1SCorey Minyard DeviceClass *dc = DEVICE_CLASS(klass); 306*5e9ae4b1SCorey Minyard I2CSlaveClass *k = I2C_SLAVE_CLASS(klass); 307*5e9ae4b1SCorey Minyard 308*5e9ae4b1SCorey Minyard dc->realize = tmp105_realize; 309*5e9ae4b1SCorey Minyard k->event = tmp105_event; 310*5e9ae4b1SCorey Minyard k->recv = tmp105_rx; 311*5e9ae4b1SCorey Minyard k->send = tmp105_tx; 312*5e9ae4b1SCorey Minyard dc->vmsd = &vmstate_tmp105; 313*5e9ae4b1SCorey Minyard } 314*5e9ae4b1SCorey Minyard 315*5e9ae4b1SCorey Minyard static const TypeInfo tmp105_info = { 316*5e9ae4b1SCorey Minyard .name = TYPE_TMP105, 317*5e9ae4b1SCorey Minyard .parent = TYPE_I2C_SLAVE, 318*5e9ae4b1SCorey Minyard .instance_size = sizeof(TMP105State), 319*5e9ae4b1SCorey Minyard .instance_init = tmp105_initfn, 320*5e9ae4b1SCorey Minyard .class_init = tmp105_class_init, 321*5e9ae4b1SCorey Minyard }; 322*5e9ae4b1SCorey Minyard 323*5e9ae4b1SCorey Minyard static void tmp105_register_types(void) 324*5e9ae4b1SCorey Minyard { 325*5e9ae4b1SCorey Minyard type_register_static(&tmp105_info); 326*5e9ae4b1SCorey Minyard } 327*5e9ae4b1SCorey Minyard 328*5e9ae4b1SCorey Minyard type_init(tmp105_register_types) 329