156 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			156 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C
		
	
	
	
/*
 | 
						|
 * (C) Copyright 2001
 | 
						|
 * Erik Theisen,  Wave 7 Optics, etheisen@mindspring.com.
 | 
						|
 *
 | 
						|
 * SPDX-License-Identifier:	GPL-2.0+
 | 
						|
 */
 | 
						|
 | 
						|
/*
 | 
						|
 * Dallas Semiconductor's DS1621/1631 Digital Thermometer and Thermostat.
 | 
						|
 */
 | 
						|
 | 
						|
#include <common.h>
 | 
						|
#include <i2c.h>
 | 
						|
#include <dtt.h>
 | 
						|
 | 
						|
/*
 | 
						|
 * Device code
 | 
						|
 */
 | 
						|
#define DTT_I2C_DEV_CODE 	0x48	/* Dallas Semi's DS1621 */
 | 
						|
#define DTT_READ_TEMP		0xAA
 | 
						|
#define DTT_READ_COUNTER	0xA8
 | 
						|
#define DTT_READ_SLOPE		0xA9
 | 
						|
#define DTT_WRITE_START_CONV	0xEE
 | 
						|
#define DTT_WRITE_STOP_CONV	0x22
 | 
						|
#define DTT_TEMP_HIGH		0xA1
 | 
						|
#define DTT_TEMP_LOW		0xA2
 | 
						|
#define DTT_CONFIG		0xAC
 | 
						|
 | 
						|
/*
 | 
						|
 * Config register bits
 | 
						|
 */
 | 
						|
#define DTT_CONFIG_1SHOT	0x01
 | 
						|
#define DTT_CONFIG_POLARITY	0x02
 | 
						|
#define DTT_CONFIG_R0		0x04	/* ds1631 only */
 | 
						|
#define DTT_CONFIG_R1		0x08	/* ds1631 only */
 | 
						|
#define DTT_CONFIG_NVB		0x10
 | 
						|
#define DTT_CONFIG_TLF		0x20
 | 
						|
#define DTT_CONFIG_THF		0x40
 | 
						|
#define DTT_CONFIG_DONE		0x80
 | 
						|
 | 
						|
 | 
						|
int dtt_read(int sensor, int reg)
 | 
						|
{
 | 
						|
	int dlen;
 | 
						|
	uchar data[2];
 | 
						|
 | 
						|
	/* Calculate sensor address and command */
 | 
						|
	sensor = DTT_I2C_DEV_CODE + (sensor & 0x07); /* Calculate addr of ds1621*/
 | 
						|
 | 
						|
	/* Prepare to handle 2 byte result */
 | 
						|
	switch(reg) {
 | 
						|
	case DTT_READ_TEMP:
 | 
						|
	case DTT_TEMP_HIGH:
 | 
						|
	case DTT_TEMP_LOW:
 | 
						|
		dlen = 2;
 | 
						|
		break;
 | 
						|
	default:
 | 
						|
		dlen = 1;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Now try to read the register */
 | 
						|
	if (i2c_read(sensor, reg, 1, data, dlen) != 0)
 | 
						|
		return 1;
 | 
						|
 | 
						|
	/* Handle 2 byte result */
 | 
						|
	if (dlen == 2)
 | 
						|
		return (short)((data[0] << 8) | data[1]);
 | 
						|
 | 
						|
	return (int)data[0];
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int dtt_write(int sensor, int reg, int val)
 | 
						|
{
 | 
						|
	int dlen;
 | 
						|
	uchar data[2];
 | 
						|
 | 
						|
	/* Calculate sensor address and register */
 | 
						|
	sensor = DTT_I2C_DEV_CODE + (sensor & 0x07);
 | 
						|
 | 
						|
	/* Handle various data sizes. */
 | 
						|
	switch(reg) {
 | 
						|
	case DTT_READ_TEMP:
 | 
						|
	case DTT_TEMP_HIGH:
 | 
						|
	case DTT_TEMP_LOW:
 | 
						|
		dlen = 2;
 | 
						|
		data[0] = (char)((val >> 8) & 0xff);	/* MSB first */
 | 
						|
		data[1] = (char)(val & 0xff);
 | 
						|
		break;
 | 
						|
	case DTT_WRITE_START_CONV:
 | 
						|
	case DTT_WRITE_STOP_CONV:
 | 
						|
		dlen = 0;
 | 
						|
		data[0] = (char)0;
 | 
						|
		data[1] = (char)0;
 | 
						|
		break;
 | 
						|
	default:
 | 
						|
		dlen = 1;
 | 
						|
		data[0] = (char)(val & 0xff);
 | 
						|
	}
 | 
						|
 | 
						|
	/* Write value to device */
 | 
						|
	if (i2c_write(sensor, reg, 1, data, dlen) != 0)
 | 
						|
		return 1;
 | 
						|
 | 
						|
	/* Poll NV memory busy bit in case write was to register stored in EEPROM */
 | 
						|
	while(i2c_reg_read(sensor, DTT_CONFIG) & DTT_CONFIG_NVB)
 | 
						|
		;
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int dtt_init_one(int sensor)
 | 
						|
{
 | 
						|
	int val;
 | 
						|
 | 
						|
	/* Setup High Temp */
 | 
						|
	val = ((CONFIG_SYS_DTT_MAX_TEMP * 2) << 7) & 0xff80;
 | 
						|
	if (dtt_write(sensor, DTT_TEMP_HIGH, val) != 0)
 | 
						|
		return 1;
 | 
						|
 | 
						|
	/* Setup Low Temp - hysteresis */
 | 
						|
	val = (((CONFIG_SYS_DTT_MAX_TEMP - CONFIG_SYS_DTT_HYSTERESIS) * 2) << 7) & 0xff80;
 | 
						|
	if (dtt_write(sensor, DTT_TEMP_LOW, val) != 0)
 | 
						|
		return 1;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Setup configuraton register
 | 
						|
	 *
 | 
						|
	 * Clear THF & TLF, Reserved = 1, Polarity = Active Low, One Shot = YES
 | 
						|
	 *
 | 
						|
	 * We run in polled mode, since there isn't any way to know if this
 | 
						|
	 * lousy device is ready to provide temperature readings on power up.
 | 
						|
	 */
 | 
						|
	val = 0x9;
 | 
						|
	if (dtt_write(sensor, DTT_CONFIG, val) != 0)
 | 
						|
		return 1;
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int dtt_get_temp(int sensor)
 | 
						|
{
 | 
						|
	int i;
 | 
						|
 | 
						|
	/* Start a conversion, may take up to 1 second. */
 | 
						|
	dtt_write(sensor, DTT_WRITE_START_CONV, 0);
 | 
						|
	for (i = 0; i <= 10; i++) {
 | 
						|
		udelay(100000);
 | 
						|
		if (dtt_read(sensor, DTT_CONFIG) & DTT_CONFIG_DONE)
 | 
						|
			break;
 | 
						|
	}
 | 
						|
 | 
						|
	return (dtt_read(sensor, DTT_READ_TEMP) / 256);
 | 
						|
}
 |