xref: /rk3399_rockchip-uboot/drivers/i2c/mvtwsi.c (revision 306563a773f1111c6beff8d1855d5b2a2137aebd)
101ec99d9SAlbert Aribaud /*
2*306563a7SAlbert Aribaud  * Driver for the TWSI (i2c) controller found on the Marvell
3*306563a7SAlbert Aribaud  * orion5x and kirkwood SoC families.
401ec99d9SAlbert Aribaud  *
5*306563a7SAlbert Aribaud  * Author: Albert Aribaud <albert.aribaud@free.fr>
6*306563a7SAlbert Aribaud  * Copyright (c) 2010 Albert Aribaud.
701ec99d9SAlbert Aribaud  *
801ec99d9SAlbert Aribaud  * See file CREDITS for list of people who contributed to this
901ec99d9SAlbert Aribaud  * project.
1001ec99d9SAlbert Aribaud  *
1101ec99d9SAlbert Aribaud  * This program is free software; you can redistribute it and/or
1201ec99d9SAlbert Aribaud  * modify it under the terms of the GNU General Public License as
1301ec99d9SAlbert Aribaud  * published by the Free Software Foundation; either version 2 of
1401ec99d9SAlbert Aribaud  * the License, or (at your option) any later version.
1501ec99d9SAlbert Aribaud  *
1601ec99d9SAlbert Aribaud  * This program is distributed in the hope that it will be useful,
1701ec99d9SAlbert Aribaud  * but WITHOUT ANY WARRANTY; without even the implied warranty of
1801ec99d9SAlbert Aribaud  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
1901ec99d9SAlbert Aribaud  * GNU General Public License for more details.
2001ec99d9SAlbert Aribaud  *
2101ec99d9SAlbert Aribaud  * You should have received a copy of the GNU General Public License
2201ec99d9SAlbert Aribaud  * along with this program; if not, write to the Free Software
2301ec99d9SAlbert Aribaud  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
2401ec99d9SAlbert Aribaud  * MA 02110-1301 USA
2501ec99d9SAlbert Aribaud  */
26*306563a7SAlbert Aribaud 
2701ec99d9SAlbert Aribaud #include <common.h>
2801ec99d9SAlbert Aribaud #include <i2c.h>
2901ec99d9SAlbert Aribaud #include <asm/errno.h>
3001ec99d9SAlbert Aribaud #include <asm/io.h>
3101ec99d9SAlbert Aribaud 
32*306563a7SAlbert Aribaud /*
33*306563a7SAlbert Aribaud  * include a file that will provide CONFIG_I2C_MVTWSI_BASE
34*306563a7SAlbert Aribaud  * and possibly other settings
35*306563a7SAlbert Aribaud  */
3601ec99d9SAlbert Aribaud 
37*306563a7SAlbert Aribaud #if defined(CONFIG_ORION5X)
38*306563a7SAlbert Aribaud #include <asm/arch/orion5x.h>
39*306563a7SAlbert Aribaud #elif defined(CONFIG_KIRKWOOD)
40*306563a7SAlbert Aribaud #include <asm/arch/kirkwood.h>
41*306563a7SAlbert Aribaud #else
42*306563a7SAlbert Aribaud #error Driver mvtwsi not supported by SoC or board
4301ec99d9SAlbert Aribaud #endif
4401ec99d9SAlbert Aribaud 
4501ec99d9SAlbert Aribaud /*
46*306563a7SAlbert Aribaud  * TWSI register structure
4701ec99d9SAlbert Aribaud  */
4801ec99d9SAlbert Aribaud 
49*306563a7SAlbert Aribaud struct  mvtwsi_registers {
50*306563a7SAlbert Aribaud 	u32 slave_address;
51*306563a7SAlbert Aribaud 	u32 data;
52*306563a7SAlbert Aribaud 	u32 control;
53*306563a7SAlbert Aribaud 	union {
54*306563a7SAlbert Aribaud 		u32 status;	/* when reading */
55*306563a7SAlbert Aribaud 		u32 baudrate;	/* when writing */
56*306563a7SAlbert Aribaud 	};
57*306563a7SAlbert Aribaud 	u32 xtnd_slave_addr;
58*306563a7SAlbert Aribaud 	u32 reserved[2];
59*306563a7SAlbert Aribaud 	u32 soft_reset;
60*306563a7SAlbert Aribaud };
61*306563a7SAlbert Aribaud 
62*306563a7SAlbert Aribaud /*
63*306563a7SAlbert Aribaud  * Control register fields
64*306563a7SAlbert Aribaud  */
65*306563a7SAlbert Aribaud 
66*306563a7SAlbert Aribaud #define	MVTWSI_CONTROL_ACK	0x00000004
67*306563a7SAlbert Aribaud #define	MVTWSI_CONTROL_IFLG	0x00000008
68*306563a7SAlbert Aribaud #define	MVTWSI_CONTROL_STOP	0x00000010
69*306563a7SAlbert Aribaud #define	MVTWSI_CONTROL_START	0x00000020
70*306563a7SAlbert Aribaud #define	MVTWSI_CONTROL_TWSIEN	0x00000040
71*306563a7SAlbert Aribaud #define	MVTWSI_CONTROL_INTEN	0x00000080
72*306563a7SAlbert Aribaud 
73*306563a7SAlbert Aribaud /*
74*306563a7SAlbert Aribaud  * Status register values -- only those expected in normal master
75*306563a7SAlbert Aribaud  * operation on non-10-bit-address devices; whatever status we don't
76*306563a7SAlbert Aribaud  * expect in nominal conditions (bus errors, arbitration losses,
77*306563a7SAlbert Aribaud  * missing ACKs...) we just pass back to the caller as an error
78*306563a7SAlbert Aribaud  * code.
79*306563a7SAlbert Aribaud  */
80*306563a7SAlbert Aribaud 
81*306563a7SAlbert Aribaud #define	MVTWSI_STATUS_START		0x08
82*306563a7SAlbert Aribaud #define	MVTWSI_STATUS_REPEATED_START	0x10
83*306563a7SAlbert Aribaud #define	MVTWSI_STATUS_ADDR_W_ACK	0x18
84*306563a7SAlbert Aribaud #define	MVTWSI_STATUS_DATA_W_ACK	0x28
85*306563a7SAlbert Aribaud #define	MVTWSI_STATUS_ADDR_R_ACK	0x40
86*306563a7SAlbert Aribaud #define	MVTWSI_STATUS_ADDR_R_NAK	0x48
87*306563a7SAlbert Aribaud #define	MVTWSI_STATUS_DATA_R_ACK	0x50
88*306563a7SAlbert Aribaud #define	MVTWSI_STATUS_DATA_R_NAK	0x58
89*306563a7SAlbert Aribaud #define	MVTWSI_STATUS_IDLE		0xF8
90*306563a7SAlbert Aribaud 
91*306563a7SAlbert Aribaud /*
92*306563a7SAlbert Aribaud  * The single instance of the controller we'll be dealing with
93*306563a7SAlbert Aribaud  */
94*306563a7SAlbert Aribaud 
95*306563a7SAlbert Aribaud static struct  mvtwsi_registers *twsi =
96*306563a7SAlbert Aribaud 	(struct  mvtwsi_registers *) CONFIG_I2C_MVTWSI_BASE;
97*306563a7SAlbert Aribaud 
98*306563a7SAlbert Aribaud /*
99*306563a7SAlbert Aribaud  * Returned statuses are 0 for success and nonzero otherwise.
100*306563a7SAlbert Aribaud  * Currently, cmd_i2c and cmd_eeprom do not interpret an error status.
101*306563a7SAlbert Aribaud  * Thus to ease debugging, the return status contains some debug info:
102*306563a7SAlbert Aribaud  * - bits 31..24 are error class: 1 is timeout, 2 is 'status mismatch'.
103*306563a7SAlbert Aribaud  * - bits 23..16 are the last value of the control register.
104*306563a7SAlbert Aribaud  * - bits 15..8 are the last value of the status register.
105*306563a7SAlbert Aribaud  * - bits 7..0 are the expected value of the status register.
106*306563a7SAlbert Aribaud  */
107*306563a7SAlbert Aribaud 
108*306563a7SAlbert Aribaud #define MVTWSI_ERROR_WRONG_STATUS	0x01
109*306563a7SAlbert Aribaud #define MVTWSI_ERROR_TIMEOUT		0x02
110*306563a7SAlbert Aribaud 
111*306563a7SAlbert Aribaud #define MVTWSI_ERROR(ec, lc, ls, es) (((ec << 24) & 0xFF000000) | \
112*306563a7SAlbert Aribaud 	((lc << 16) & 0x00FF0000) | ((ls<<8) & 0x0000FF00) | (es & 0xFF))
113*306563a7SAlbert Aribaud 
114*306563a7SAlbert Aribaud /*
115*306563a7SAlbert Aribaud  * Wait for IFLG to raise, or return 'timeout'; then if status is as expected,
116*306563a7SAlbert Aribaud  * return 0 (ok) or return 'wrong status'.
117*306563a7SAlbert Aribaud  */
118*306563a7SAlbert Aribaud static int twsi_wait(int expected_status)
11901ec99d9SAlbert Aribaud {
120*306563a7SAlbert Aribaud 	int control, status;
121*306563a7SAlbert Aribaud 	int timeout = 1000;
122*306563a7SAlbert Aribaud 
123*306563a7SAlbert Aribaud 	do {
124*306563a7SAlbert Aribaud 		control = readl(&twsi->control);
125*306563a7SAlbert Aribaud 		if (control & MVTWSI_CONTROL_IFLG) {
126*306563a7SAlbert Aribaud 			status = readl(&twsi->status);
127*306563a7SAlbert Aribaud 			if (status == expected_status)
128*306563a7SAlbert Aribaud 				return 0;
12901ec99d9SAlbert Aribaud 			else
130*306563a7SAlbert Aribaud 				return MVTWSI_ERROR(
131*306563a7SAlbert Aribaud 					MVTWSI_ERROR_WRONG_STATUS,
132*306563a7SAlbert Aribaud 					control, status, expected_status);
133*306563a7SAlbert Aribaud 		}
134*306563a7SAlbert Aribaud 		udelay(10); /* one clock cycle at 100 kHz */
135*306563a7SAlbert Aribaud 	} while (timeout--);
136*306563a7SAlbert Aribaud 	status = readl(&twsi->status);
137*306563a7SAlbert Aribaud 	return MVTWSI_ERROR(
138*306563a7SAlbert Aribaud 		MVTWSI_ERROR_TIMEOUT, control, status, expected_status);
13901ec99d9SAlbert Aribaud }
14001ec99d9SAlbert Aribaud 
141*306563a7SAlbert Aribaud /*
142*306563a7SAlbert Aribaud  * These flags are ORed to any write to the control register
143*306563a7SAlbert Aribaud  * They allow global setting of TWSIEN and ACK.
144*306563a7SAlbert Aribaud  * By default none are set.
145*306563a7SAlbert Aribaud  * twsi_start() sets TWSIEN (in case the controller was disabled)
146*306563a7SAlbert Aribaud  * twsi_recv() sets ACK or resets it depending on expected status.
147*306563a7SAlbert Aribaud  */
148*306563a7SAlbert Aribaud static u8 twsi_control_flags = MVTWSI_CONTROL_TWSIEN;
14901ec99d9SAlbert Aribaud 
150*306563a7SAlbert Aribaud /*
151*306563a7SAlbert Aribaud  * Assert the START condition, either in a single I2C transaction
152*306563a7SAlbert Aribaud  * or inside back-to-back ones (repeated starts).
153*306563a7SAlbert Aribaud  */
154*306563a7SAlbert Aribaud static int twsi_start(int expected_status)
155*306563a7SAlbert Aribaud {
156*306563a7SAlbert Aribaud 	/* globally set TWSIEN in case it was not */
157*306563a7SAlbert Aribaud 	twsi_control_flags |= MVTWSI_CONTROL_TWSIEN;
158*306563a7SAlbert Aribaud 	/* assert START */
159*306563a7SAlbert Aribaud 	writel(twsi_control_flags | MVTWSI_CONTROL_START, &twsi->control);
160*306563a7SAlbert Aribaud 	/* wait for controller to process START */
161*306563a7SAlbert Aribaud 	return twsi_wait(expected_status);
162*306563a7SAlbert Aribaud }
163*306563a7SAlbert Aribaud 
164*306563a7SAlbert Aribaud /*
165*306563a7SAlbert Aribaud  * Send a byte (i2c address or data).
166*306563a7SAlbert Aribaud  */
167*306563a7SAlbert Aribaud static int twsi_send(u8 byte, int expected_status)
168*306563a7SAlbert Aribaud {
169*306563a7SAlbert Aribaud 	/* put byte in data register for sending */
170*306563a7SAlbert Aribaud 	writel(byte, &twsi->data);
171*306563a7SAlbert Aribaud 	/* clear any pending interrupt -- that'll cause sending */
172*306563a7SAlbert Aribaud 	writel(twsi_control_flags, &twsi->control);
173*306563a7SAlbert Aribaud 	/* wait for controller to receive byte and check ACK */
174*306563a7SAlbert Aribaud 	return twsi_wait(expected_status);
175*306563a7SAlbert Aribaud }
176*306563a7SAlbert Aribaud 
177*306563a7SAlbert Aribaud /*
178*306563a7SAlbert Aribaud  * Receive a byte.
179*306563a7SAlbert Aribaud  * Global mvtwsi_control_flags variable says if we should ack or nak.
180*306563a7SAlbert Aribaud  */
181*306563a7SAlbert Aribaud static int twsi_recv(u8 *byte)
182*306563a7SAlbert Aribaud {
183*306563a7SAlbert Aribaud 	int expected_status, status;
184*306563a7SAlbert Aribaud 
185*306563a7SAlbert Aribaud 	/* compute expected status based on ACK bit in global control flags */
186*306563a7SAlbert Aribaud 	if (twsi_control_flags & MVTWSI_CONTROL_ACK)
187*306563a7SAlbert Aribaud 		expected_status = MVTWSI_STATUS_DATA_R_ACK;
188*306563a7SAlbert Aribaud 	else
189*306563a7SAlbert Aribaud 		expected_status = MVTWSI_STATUS_DATA_R_NAK;
190*306563a7SAlbert Aribaud 	/* acknowledge *previous state* and launch receive */
191*306563a7SAlbert Aribaud 	writel(twsi_control_flags, &twsi->control);
192*306563a7SAlbert Aribaud 	/* wait for controller to receive byte and assert ACK or NAK */
193*306563a7SAlbert Aribaud 	status = twsi_wait(expected_status);
194*306563a7SAlbert Aribaud 	/* if we did receive expected byte then store it */
195*306563a7SAlbert Aribaud 	if (status == 0)
196*306563a7SAlbert Aribaud 		*byte = readl(&twsi->data);
197*306563a7SAlbert Aribaud 	/* return status */
198*306563a7SAlbert Aribaud 	return status;
199*306563a7SAlbert Aribaud }
200*306563a7SAlbert Aribaud 
201*306563a7SAlbert Aribaud /*
202*306563a7SAlbert Aribaud  * Assert the STOP condition.
203*306563a7SAlbert Aribaud  * This is also used to force the bus back in idle (SDA=SCL=1).
204*306563a7SAlbert Aribaud  */
205*306563a7SAlbert Aribaud static int twsi_stop(int status)
206*306563a7SAlbert Aribaud {
207*306563a7SAlbert Aribaud 	int control, stop_status;
208*306563a7SAlbert Aribaud 	int timeout = 1000;
209*306563a7SAlbert Aribaud 
210*306563a7SAlbert Aribaud 	/* assert STOP */
211*306563a7SAlbert Aribaud 	control = MVTWSI_CONTROL_TWSIEN | MVTWSI_CONTROL_STOP;
212*306563a7SAlbert Aribaud 	writel(control, &twsi->control);
213*306563a7SAlbert Aribaud 	/* wait for IDLE; IFLG won't rise so twsi_wait() is no use. */
214*306563a7SAlbert Aribaud 	do {
215*306563a7SAlbert Aribaud 		stop_status = readl(&twsi->status);
216*306563a7SAlbert Aribaud 		if (stop_status == MVTWSI_STATUS_IDLE)
217*306563a7SAlbert Aribaud 			break;
218*306563a7SAlbert Aribaud 		udelay(10); /* one clock cycle at 100 kHz */
219*306563a7SAlbert Aribaud 	} while (timeout--);
220*306563a7SAlbert Aribaud 	control = readl(&twsi->control);
221*306563a7SAlbert Aribaud 	if (stop_status != MVTWSI_STATUS_IDLE)
222*306563a7SAlbert Aribaud 		if (status == 0)
223*306563a7SAlbert Aribaud 			status = MVTWSI_ERROR(
224*306563a7SAlbert Aribaud 				MVTWSI_ERROR_TIMEOUT,
225*306563a7SAlbert Aribaud 				control, status, MVTWSI_STATUS_IDLE);
226*306563a7SAlbert Aribaud 	return status;
227*306563a7SAlbert Aribaud }
228*306563a7SAlbert Aribaud 
229*306563a7SAlbert Aribaud /*
230*306563a7SAlbert Aribaud  * Ugly formula to convert m and n values to a frequency comes from
231*306563a7SAlbert Aribaud  * TWSI specifications
232*306563a7SAlbert Aribaud  */
233*306563a7SAlbert Aribaud 
234*306563a7SAlbert Aribaud #define TWSI_FREQUENCY(m, n) \
235*306563a7SAlbert Aribaud 	((u8) (CONFIG_SYS_TCLK / (10 * (m + 1) * 2 * (1 << n))))
236*306563a7SAlbert Aribaud 
237*306563a7SAlbert Aribaud /*
238*306563a7SAlbert Aribaud  * These are required to be reprogrammed before enabling the controller
239*306563a7SAlbert Aribaud  * because a reset loses them.
240*306563a7SAlbert Aribaud  * Default values come from the spec, but a twsi_reset will change them.
241*306563a7SAlbert Aribaud  * twsi_slave_address left uninitialized lest checkpatch.pl complains.
242*306563a7SAlbert Aribaud  */
243*306563a7SAlbert Aribaud 
244*306563a7SAlbert Aribaud /* Baudrate generator: m (bits 7..4) =4, n (bits 3..0) =4 */
245*306563a7SAlbert Aribaud static u8 twsi_baud_rate = 0x44; /* baudrate at controller reset */
246*306563a7SAlbert Aribaud /* Default frequency corresponding to default m=4, n=4 */
247*306563a7SAlbert Aribaud static u8 twsi_actual_speed = TWSI_FREQUENCY(4, 4);
248*306563a7SAlbert Aribaud /* Default slave address is 0 (so is an uninitialized static) */
249*306563a7SAlbert Aribaud static u8 twsi_slave_address;
250*306563a7SAlbert Aribaud 
251*306563a7SAlbert Aribaud /*
252*306563a7SAlbert Aribaud  * Reset controller.
253*306563a7SAlbert Aribaud  * Called at end of i2c_init unsuccessful i2c transactions.
254*306563a7SAlbert Aribaud  * Controller reset also resets the baud rate and slave address, so
255*306563a7SAlbert Aribaud  * re-establish them.
256*306563a7SAlbert Aribaud  */
257*306563a7SAlbert Aribaud static void twsi_reset(void)
258*306563a7SAlbert Aribaud {
259*306563a7SAlbert Aribaud 	/* ensure controller will be enabled by any twsi*() function */
260*306563a7SAlbert Aribaud 	twsi_control_flags = MVTWSI_CONTROL_TWSIEN;
261*306563a7SAlbert Aribaud 	/* reset controller */
262*306563a7SAlbert Aribaud 	writel(0, &twsi->soft_reset);
263*306563a7SAlbert Aribaud 	/* wait 2 ms -- this is what the Marvell LSP does */
264*306563a7SAlbert Aribaud 	udelay(20000);
265*306563a7SAlbert Aribaud 	/* set baud rate */
266*306563a7SAlbert Aribaud 	writel(twsi_baud_rate, &twsi->baudrate);
267*306563a7SAlbert Aribaud 	/* set slave address even though we don't use it */
268*306563a7SAlbert Aribaud 	writel(twsi_slave_address, &twsi->slave_address);
269*306563a7SAlbert Aribaud 	writel(0, &twsi->xtnd_slave_addr);
270*306563a7SAlbert Aribaud 	/* assert STOP but don't care for the result */
271*306563a7SAlbert Aribaud 	(void) twsi_stop(0);
272*306563a7SAlbert Aribaud }
273*306563a7SAlbert Aribaud 
274*306563a7SAlbert Aribaud /*
275*306563a7SAlbert Aribaud  * I2C init called by cmd_i2c when doing 'i2c reset'.
276*306563a7SAlbert Aribaud  * Sets baud to the highest possible value not exceeding requested one.
277*306563a7SAlbert Aribaud  */
278*306563a7SAlbert Aribaud void i2c_init(int requested_speed, int slaveadd)
279*306563a7SAlbert Aribaud {
280*306563a7SAlbert Aribaud 	int	tmp_speed, highest_speed, n, m;
281*306563a7SAlbert Aribaud 	int	baud = 0x44; /* baudrate at controller reset */
282*306563a7SAlbert Aribaud 
283*306563a7SAlbert Aribaud 	/* use actual speed to collect progressively higher values */
284*306563a7SAlbert Aribaud 	highest_speed = 0;
285*306563a7SAlbert Aribaud 	/* compute m, n setting for highest speed not above requested speed */
28601ec99d9SAlbert Aribaud 	for (n = 0; n < 8; n++) {
28701ec99d9SAlbert Aribaud 		for (m = 0; m < 16; m++) {
288*306563a7SAlbert Aribaud 			tmp_speed = TWSI_FREQUENCY(m, n);
289*306563a7SAlbert Aribaud 			if ((tmp_speed <= requested_speed)
290*306563a7SAlbert Aribaud 			 && (tmp_speed > highest_speed)) {
291*306563a7SAlbert Aribaud 				highest_speed = tmp_speed;
292*306563a7SAlbert Aribaud 				baud = (m << 3) | n;
29301ec99d9SAlbert Aribaud 			}
29401ec99d9SAlbert Aribaud 		}
29501ec99d9SAlbert Aribaud 	}
296*306563a7SAlbert Aribaud 	/* save baud rate and slave for later calls to twsi_reset */
297*306563a7SAlbert Aribaud 	twsi_baud_rate = baud;
298*306563a7SAlbert Aribaud 	twsi_actual_speed = highest_speed;
299*306563a7SAlbert Aribaud 	twsi_slave_address = slaveadd;
300*306563a7SAlbert Aribaud 	/* reset controller */
301*306563a7SAlbert Aribaud 	twsi_reset();
30201ec99d9SAlbert Aribaud }
30301ec99d9SAlbert Aribaud 
30401ec99d9SAlbert Aribaud /*
305*306563a7SAlbert Aribaud  * Begin I2C transaction with expected start status, at given address.
306*306563a7SAlbert Aribaud  * Common to i2c_probe, i2c_read and i2c_write.
307*306563a7SAlbert Aribaud  * Expected address status will derive from direction bit (bit 0) in addr.
30801ec99d9SAlbert Aribaud  */
309*306563a7SAlbert Aribaud static int i2c_begin(int expected_start_status, u8 addr)
31001ec99d9SAlbert Aribaud {
311*306563a7SAlbert Aribaud 	int status, expected_addr_status;
31201ec99d9SAlbert Aribaud 
313*306563a7SAlbert Aribaud 	/* compute expected address status from direction bit in addr */
314*306563a7SAlbert Aribaud 	if (addr & 1) /* reading */
315*306563a7SAlbert Aribaud 		expected_addr_status = MVTWSI_STATUS_ADDR_R_ACK;
316*306563a7SAlbert Aribaud 	else /* writing */
317*306563a7SAlbert Aribaud 		expected_addr_status = MVTWSI_STATUS_ADDR_W_ACK;
318*306563a7SAlbert Aribaud 	/* assert START */
319*306563a7SAlbert Aribaud 	status = twsi_start(expected_start_status);
320*306563a7SAlbert Aribaud 	/* send out the address if the start went well */
321*306563a7SAlbert Aribaud 	if (status == 0)
322*306563a7SAlbert Aribaud 		status = twsi_send(addr, expected_addr_status);
323*306563a7SAlbert Aribaud 	/* return ok or status of first failure to caller */
324*306563a7SAlbert Aribaud 	return status;
32501ec99d9SAlbert Aribaud }
32601ec99d9SAlbert Aribaud 
327*306563a7SAlbert Aribaud /*
328*306563a7SAlbert Aribaud  * I2C probe called by cmd_i2c when doing 'i2c probe'.
329*306563a7SAlbert Aribaud  * Begin read, nak data byte, end.
330*306563a7SAlbert Aribaud  */
331*306563a7SAlbert Aribaud int i2c_probe(uchar chip)
33201ec99d9SAlbert Aribaud {
333*306563a7SAlbert Aribaud 	u8 dummy_byte;
334*306563a7SAlbert Aribaud 	int status;
33501ec99d9SAlbert Aribaud 
336*306563a7SAlbert Aribaud 	/* begin i2c read */
337*306563a7SAlbert Aribaud 	status = i2c_begin(MVTWSI_STATUS_START, (chip << 1) | 1);
338*306563a7SAlbert Aribaud 	/* dummy read was accepted: receive byte but NAK it. */
339*306563a7SAlbert Aribaud 	if (status == 0)
340*306563a7SAlbert Aribaud 		status = twsi_recv(&dummy_byte);
341*306563a7SAlbert Aribaud 	/* Stop transaction */
342*306563a7SAlbert Aribaud 	twsi_stop(0);
343*306563a7SAlbert Aribaud 	/* return 0 or status of first failure */
344*306563a7SAlbert Aribaud 	return status;
34501ec99d9SAlbert Aribaud }
34601ec99d9SAlbert Aribaud 
347*306563a7SAlbert Aribaud /*
348*306563a7SAlbert Aribaud  * I2C read called by cmd_i2c when doing 'i2c read' and by cmd_eeprom.c
349*306563a7SAlbert Aribaud  * Begin write, send address byte(s), begin read, receive data bytes, end.
350*306563a7SAlbert Aribaud  *
351*306563a7SAlbert Aribaud  * NOTE: some EEPROMS want a stop right before the second start, while
352*306563a7SAlbert Aribaud  * some will choke if it is there. Deciding which we should do is eeprom
353*306563a7SAlbert Aribaud  * stuff, not i2c, but at the moment the APIs won't let us put it in
354*306563a7SAlbert Aribaud  * cmd_eeprom, so we have to choose here, and for the moment that'll be
355*306563a7SAlbert Aribaud  * a repeated start without a preceding stop.
356*306563a7SAlbert Aribaud  */
357*306563a7SAlbert Aribaud int i2c_read(u8 dev, uint addr, int alen, u8 *data, int length)
35801ec99d9SAlbert Aribaud {
359*306563a7SAlbert Aribaud 	int status;
36001ec99d9SAlbert Aribaud 
361*306563a7SAlbert Aribaud 	/* begin i2c write to send the address bytes */
362*306563a7SAlbert Aribaud 	status = i2c_begin(MVTWSI_STATUS_START, (dev << 1));
363*306563a7SAlbert Aribaud 	/* send addr bytes */
364*306563a7SAlbert Aribaud 	while ((status == 0) && alen--)
365*306563a7SAlbert Aribaud 		status = twsi_send(addr >> (8*alen),
366*306563a7SAlbert Aribaud 			MVTWSI_STATUS_DATA_W_ACK);
367*306563a7SAlbert Aribaud 	/* begin i2c read to receive eeprom data bytes */
368*306563a7SAlbert Aribaud 	if (status == 0)
369*306563a7SAlbert Aribaud 		status = i2c_begin(
370*306563a7SAlbert Aribaud 			MVTWSI_STATUS_REPEATED_START, (dev << 1) | 1);
371*306563a7SAlbert Aribaud 	/* prepare ACK if at least one byte must be received */
372*306563a7SAlbert Aribaud 	if (length > 0)
373*306563a7SAlbert Aribaud 		twsi_control_flags |= MVTWSI_CONTROL_ACK;
374*306563a7SAlbert Aribaud 	/* now receive actual bytes */
375*306563a7SAlbert Aribaud 	while ((status == 0) && length--) {
376*306563a7SAlbert Aribaud 		/* reset NAK if we if no more to read now */
377*306563a7SAlbert Aribaud 		if (length == 0)
378*306563a7SAlbert Aribaud 			twsi_control_flags &= ~MVTWSI_CONTROL_ACK;
379*306563a7SAlbert Aribaud 		/* read current byte */
380*306563a7SAlbert Aribaud 		status = twsi_recv(data++);
38101ec99d9SAlbert Aribaud 	}
382*306563a7SAlbert Aribaud 	/* Stop transaction */
383*306563a7SAlbert Aribaud 	status = twsi_stop(status);
384*306563a7SAlbert Aribaud 	/* return 0 or status of first failure */
385*306563a7SAlbert Aribaud 	return status;
38601ec99d9SAlbert Aribaud }
38701ec99d9SAlbert Aribaud 
388*306563a7SAlbert Aribaud /*
389*306563a7SAlbert Aribaud  * I2C write called by cmd_i2c when doing 'i2c write' and by cmd_eeprom.c
390*306563a7SAlbert Aribaud  * Begin write, send address byte(s), send data bytes, end.
391*306563a7SAlbert Aribaud  */
392*306563a7SAlbert Aribaud int i2c_write(u8 dev, uint addr, int alen, u8 *data, int length)
39301ec99d9SAlbert Aribaud {
394*306563a7SAlbert Aribaud 	int status;
39501ec99d9SAlbert Aribaud 
396*306563a7SAlbert Aribaud 	/* begin i2c write to send the eeprom adress bytes then data bytes */
397*306563a7SAlbert Aribaud 	status = i2c_begin(MVTWSI_STATUS_START, (dev << 1));
398*306563a7SAlbert Aribaud 	/* send addr bytes */
399*306563a7SAlbert Aribaud 	while ((status == 0) && alen--)
400*306563a7SAlbert Aribaud 		status = twsi_send(addr >> (8*alen),
401*306563a7SAlbert Aribaud 			MVTWSI_STATUS_DATA_W_ACK);
402*306563a7SAlbert Aribaud 	/* send data bytes */
403*306563a7SAlbert Aribaud 	while ((status == 0) && (length-- > 0))
404*306563a7SAlbert Aribaud 		status = twsi_send(*(data++), MVTWSI_STATUS_DATA_W_ACK);
405*306563a7SAlbert Aribaud 	/* Stop transaction */
406*306563a7SAlbert Aribaud 	status = twsi_stop(status);
407*306563a7SAlbert Aribaud 	/* return 0 or status of first failure */
408*306563a7SAlbert Aribaud 	return status;
40901ec99d9SAlbert Aribaud }
41001ec99d9SAlbert Aribaud 
411*306563a7SAlbert Aribaud /*
412*306563a7SAlbert Aribaud  * Bus set routine: we only support bus 0.
413*306563a7SAlbert Aribaud  */
41401ec99d9SAlbert Aribaud int i2c_set_bus_num(unsigned int bus)
41501ec99d9SAlbert Aribaud {
41601ec99d9SAlbert Aribaud 	if (bus > 0) {
41701ec99d9SAlbert Aribaud 		return -1;
41801ec99d9SAlbert Aribaud 	}
41901ec99d9SAlbert Aribaud 	return 0;
42001ec99d9SAlbert Aribaud }
42101ec99d9SAlbert Aribaud 
422*306563a7SAlbert Aribaud /*
423*306563a7SAlbert Aribaud  * Bus get routine: hard-return bus 0.
424*306563a7SAlbert Aribaud  */
42501ec99d9SAlbert Aribaud unsigned int i2c_get_bus_num(void)
42601ec99d9SAlbert Aribaud {
427*306563a7SAlbert Aribaud 	return 0;
42801ec99d9SAlbert Aribaud }
429