mcp980x.c revision 1.1
11.1Srkujawa/*	$NetBSD: mcp980x.c,v 1.1 2013/05/06 22:04:12 rkujawa Exp $ */
21.1Srkujawa
31.1Srkujawa/*-
41.1Srkujawa * Copyright (c) 2013 The NetBSD Foundation, Inc.
51.1Srkujawa * All rights reserved.
61.1Srkujawa *
71.1Srkujawa * This code is derived from software contributed to The NetBSD Foundation
81.1Srkujawa * by Radoslaw Kujawa.
91.1Srkujawa *
101.1Srkujawa * Redistribution and use in source and binary forms, with or without
111.1Srkujawa * modification, are permitted provided that the following conditions
121.1Srkujawa * are met:
131.1Srkujawa * 1. Redistributions of source code must retain the above copyright
141.1Srkujawa *    notice, this list of conditions and the following disclaimer.
151.1Srkujawa * 2. Redistributions in binary form must reproduce the above copyright
161.1Srkujawa *    notice, this list of conditions and the following disclaimer in the
171.1Srkujawa *    documentation and/or other materials provided with the distribution.
181.1Srkujawa *
191.1Srkujawa * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
201.1Srkujawa * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
211.1Srkujawa * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
221.1Srkujawa * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
231.1Srkujawa * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
241.1Srkujawa * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
251.1Srkujawa * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
261.1Srkujawa * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
271.1Srkujawa * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
281.1Srkujawa * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
291.1Srkujawa * POSSIBILITY OF SUCH DAMAGE.
301.1Srkujawa */
311.1Srkujawa
321.1Srkujawa/*
331.1Srkujawa * Microchip MCP9800/1/2/3 2-Wire High-Accuracy Temperature Sensor driver.
341.1Srkujawa * TODO: everything besides simple temperature read with default configuration.
351.1Srkujawa *
361.1Srkujawa * Note: MCP9805 is different and is supported by the sdtemp(4) driver.
371.1Srkujawa */
381.1Srkujawa
391.1Srkujawa#include <sys/cdefs.h>
401.1Srkujawa__KERNEL_RCSID(0, "$NetBSD: mcp980x.c,v 1.1 2013/05/06 22:04:12 rkujawa Exp $");
411.1Srkujawa
421.1Srkujawa#include <sys/param.h>
431.1Srkujawa#include <sys/systm.h>
441.1Srkujawa#include <sys/device.h>
451.1Srkujawa#include <sys/kernel.h>
461.1Srkujawa#include <sys/mutex.h>
471.1Srkujawa#include <sys/endian.h>
481.1Srkujawa
491.1Srkujawa#include <sys/bus.h>
501.1Srkujawa#include <dev/i2c/i2cvar.h>
511.1Srkujawa
521.1Srkujawa#include <dev/sysmon/sysmonvar.h>
531.1Srkujawa
541.1Srkujawa#include <dev/i2c/mcp980xreg.h>
551.1Srkujawa
561.1Srkujawastruct mcp980x_softc {
571.1Srkujawa	device_t		sc_dev;
581.1Srkujawa
591.1Srkujawa	i2c_tag_t		sc_tag;
601.1Srkujawa	i2c_addr_t		sc_addr;
611.1Srkujawa
621.1Srkujawa	/* envsys(4) stuff */
631.1Srkujawa	struct sysmon_envsys	*sc_sme;
641.1Srkujawa	envsys_data_t		sc_sensor;
651.1Srkujawa	kmutex_t		sc_lock;
661.1Srkujawa};
671.1Srkujawa
681.1Srkujawa
691.1Srkujawastatic int mcp980x_match(device_t, cfdata_t, void *);
701.1Srkujawastatic void mcp980x_attach(device_t, device_t, void *);
711.1Srkujawa
721.1Srkujawa/*static uint8_t mcp980x_reg_read_1(struct mcp980x_softc *sc, uint8_t);*/
731.1Srkujawastatic uint16_t mcp980x_reg_read_2(struct mcp980x_softc *sc, uint8_t reg);
741.1Srkujawa
751.1Srkujawastatic uint32_t mcp980x_temperature(struct mcp980x_softc *sc);
761.1Srkujawa
771.1Srkujawastatic void mcp980x_envsys_register(struct mcp980x_softc *);
781.1Srkujawastatic void mcp980x_envsys_refresh(struct sysmon_envsys *, envsys_data_t *);
791.1Srkujawa
801.1SrkujawaCFATTACH_DECL_NEW(mcp980x, sizeof (struct mcp980x_softc),
811.1Srkujawa    mcp980x_match, mcp980x_attach, NULL, NULL);
821.1Srkujawa
831.1Srkujawastatic int
841.1Srkujawamcp980x_match(device_t parent, cfdata_t cf, void *aux)
851.1Srkujawa{
861.1Srkujawa	/*
871.1Srkujawa	 * No sane way to probe? Perhaps at least try to match constant part
881.1Srkujawa	 * of the I2Caddress.
891.1Srkujawa	 */
901.1Srkujawa
911.1Srkujawa	return 1;
921.1Srkujawa}
931.1Srkujawa
941.1Srkujawastatic void
951.1Srkujawamcp980x_attach(device_t parent, device_t self, void *aux)
961.1Srkujawa{
971.1Srkujawa	struct mcp980x_softc *sc = device_private(self);
981.1Srkujawa	struct i2c_attach_args *ia = aux;
991.1Srkujawa
1001.1Srkujawa	sc->sc_dev = self;
1011.1Srkujawa	sc->sc_addr = ia->ia_addr;
1021.1Srkujawa	sc->sc_tag = ia->ia_tag;
1031.1Srkujawa
1041.1Srkujawa	aprint_normal(": Microchip MCP980x Temperature Sensor\n");
1051.1Srkujawa
1061.1Srkujawa	mutex_init(&sc->sc_lock, MUTEX_DEFAULT, IPL_NONE);
1071.1Srkujawa
1081.1Srkujawa	mcp980x_envsys_register(sc);
1091.1Srkujawa}
1101.1Srkujawa
1111.1Srkujawastatic uint16_t
1121.1Srkujawamcp980x_reg_read_2(struct mcp980x_softc *sc, uint8_t reg)
1131.1Srkujawa{
1141.1Srkujawa	uint8_t wbuf[2];
1151.1Srkujawa	uint16_t rv;
1161.1Srkujawa
1171.1Srkujawa	if (iic_acquire_bus(sc->sc_tag, I2C_F_POLL) != 0) {
1181.1Srkujawa		aprint_error_dev(sc->sc_dev, "cannot acquire bus for read\n");
1191.1Srkujawa		return 0;
1201.1Srkujawa	}
1211.1Srkujawa
1221.1Srkujawa	wbuf[0] = reg;
1231.1Srkujawa
1241.1Srkujawa	if (iic_exec(sc->sc_tag, I2C_OP_READ_WITH_STOP, sc->sc_addr, wbuf,
1251.1Srkujawa	    1, &rv, 2, I2C_F_POLL)) {
1261.1Srkujawa		aprint_error_dev(sc->sc_dev, "cannot execute operation\n");
1271.1Srkujawa		iic_release_bus(sc->sc_tag, I2C_F_POLL);
1281.1Srkujawa		return 0;
1291.1Srkujawa	}
1301.1Srkujawa	iic_release_bus(sc->sc_tag, I2C_F_POLL);
1311.1Srkujawa
1321.1Srkujawa	return be16toh(rv);
1331.1Srkujawa}
1341.1Srkujawa
1351.1Srkujawa/* Will need that later for reading config register. */
1361.1Srkujawa/*
1371.1Srkujawastatic uint8_t
1381.1Srkujawamcp980x_reg_read_1(struct mcp980x_softc *sc, uint8_t reg)
1391.1Srkujawa{
1401.1Srkujawa	uint8_t rv, wbuf[2];
1411.1Srkujawa
1421.1Srkujawa	if (iic_acquire_bus(sc->sc_tag, I2C_F_POLL) != 0) {
1431.1Srkujawa		aprint_error_dev(sc->sc_dev, "cannot acquire bus for read\n");
1441.1Srkujawa		return 0;
1451.1Srkujawa	}
1461.1Srkujawa
1471.1Srkujawa	wbuf[0] = reg;
1481.1Srkujawa
1491.1Srkujawa	if (iic_exec(sc->sc_tag, I2C_OP_READ_WITH_STOP, sc->sc_addr, wbuf,
1501.1Srkujawa	    1, &rv, 1, I2C_F_POLL)) {
1511.1Srkujawa		aprint_error_dev(sc->sc_dev, "cannot execute operation\n");
1521.1Srkujawa		iic_release_bus(sc->sc_tag, I2C_F_POLL);
1531.1Srkujawa		return 0;
1541.1Srkujawa	}
1551.1Srkujawa	iic_release_bus(sc->sc_tag, I2C_F_POLL);
1561.1Srkujawa
1571.1Srkujawa	return rv;
1581.1Srkujawa}*/
1591.1Srkujawa
1601.1Srkujawa/* Get temperature in microKelvins. */
1611.1Srkujawastatic uint32_t
1621.1Srkujawamcp980x_temperature(struct mcp980x_softc *sc)
1631.1Srkujawa{
1641.1Srkujawa	uint16_t raw;
1651.1Srkujawa	uint32_t rv, uk, basedegc;
1661.1Srkujawa
1671.1Srkujawa	raw = mcp980x_reg_read_2(sc, MCP980X_AMBIENT_TEMP);
1681.1Srkujawa
1691.1Srkujawa	basedegc = (raw & MCP980X_AMBIENT_TEMP_DEGREES) >>
1701.1Srkujawa	    MCP980X_AMBIENT_TEMP_DEGREES_SHIFT;
1711.1Srkujawa
1721.1Srkujawa	uk = 1000000 * basedegc;
1731.1Srkujawa
1741.1Srkujawa	if (raw & MCP980X_AMBIENT_TEMP_05DEGREE)
1751.1Srkujawa		uk += 500000;
1761.1Srkujawa	if (raw & MCP980X_AMBIENT_TEMP_025DEGREE)
1771.1Srkujawa		uk += 250000;
1781.1Srkujawa	if (raw & MCP980X_AMBIENT_TEMP_0125DEGREE)
1791.1Srkujawa		uk += 125000;
1801.1Srkujawa	if (raw & MCP980X_AMBIENT_TEMP_00625DEGREE)
1811.1Srkujawa		uk += 62500;
1821.1Srkujawa
1831.1Srkujawa	if (raw & MCP980X_AMBIENT_TEMP_SIGN)
1841.1Srkujawa		rv = 273150000U - uk;
1851.1Srkujawa	else
1861.1Srkujawa		rv = 273150000U + uk;
1871.1Srkujawa
1881.1Srkujawa	return rv;
1891.1Srkujawa}
1901.1Srkujawa
1911.1Srkujawastatic void
1921.1Srkujawamcp980x_envsys_register(struct mcp980x_softc *sc)
1931.1Srkujawa{
1941.1Srkujawa	sc->sc_sme = sysmon_envsys_create();
1951.1Srkujawa
1961.1Srkujawa	strlcpy(sc->sc_sensor.desc, "Ambient temp",
1971.1Srkujawa	    sizeof(sc->sc_sensor.desc));
1981.1Srkujawa	sc->sc_sensor.units = ENVSYS_STEMP;
1991.1Srkujawa	sc->sc_sensor.state = ENVSYS_SINVALID;
2001.1Srkujawa
2011.1Srkujawa	if (sysmon_envsys_sensor_attach(sc->sc_sme, &sc->sc_sensor)) {
2021.1Srkujawa		aprint_error_dev(sc->sc_dev,
2031.1Srkujawa		    "error attaching sensor\n");
2041.1Srkujawa		return;
2051.1Srkujawa	}
2061.1Srkujawa
2071.1Srkujawa	sc->sc_sme->sme_name = device_xname(sc->sc_dev);
2081.1Srkujawa	sc->sc_sme->sme_cookie = sc;
2091.1Srkujawa	sc->sc_sme->sme_refresh = mcp980x_envsys_refresh;
2101.1Srkujawa
2111.1Srkujawa	if (sysmon_envsys_register(sc->sc_sme)) {
2121.1Srkujawa		aprint_error_dev(sc->sc_dev, "unable to register in sysmon\n");
2131.1Srkujawa		sysmon_envsys_destroy(sc->sc_sme);
2141.1Srkujawa	}
2151.1Srkujawa}
2161.1Srkujawa
2171.1Srkujawastatic void
2181.1Srkujawamcp980x_envsys_refresh(struct sysmon_envsys *sme, envsys_data_t *edata)
2191.1Srkujawa{
2201.1Srkujawa	struct mcp980x_softc *sc = sme->sme_cookie;
2211.1Srkujawa
2221.1Srkujawa	mutex_enter(&sc->sc_lock);
2231.1Srkujawa
2241.1Srkujawa	edata->value_cur = mcp980x_temperature(sc);
2251.1Srkujawa	edata->state = ENVSYS_SVALID;
2261.1Srkujawa
2271.1Srkujawa	mutex_exit(&sc->sc_lock);
2281.1Srkujawa}
229