OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [trunk/] [rc203soc/] [sw/] [uClinux/] [arch/] [armnommu/] [kernel/] [iic.c] - Rev 1622

Go to most recent revision | Compare with Previous | Blame | View Log

/*
 * linux/arch/arm/kernel/iic.c
 *
 * Copyright (C) 1995, 1996 Russell King
 *
 * IIC is used to get the current time from the CMOS rtc.
 */
 
#include <asm/system.h>
#include <asm/delay.h>
#include <asm/io.h>
#include <asm/hardware.h>
 
#define FORCE_ONES	0xdc
 
/*
 * if delay loop has been calibrated then us that,
 * else use IOC timer 1.
 */
static void iic_delay (void)
{
	extern unsigned long loops_per_sec;
	if (loops_per_sec != (1 << 12)) {
		udelay(10);
		return;
	} else {
		unsigned long flags;
		save_flags_cli(flags);
 
		outb(254,  IOC_T1LTCHL);
		outb(255,  IOC_T1LTCHH);
		outb(0,    IOC_T1GO);
		outb(1<<6, IOC_IRQCLRA);			/* clear T1 irq */
		outb(4,    IOC_T1LTCHL);
		outb(0,    IOC_T1LTCHH);
		outb(0,    IOC_T1GO);
		while ((inb(IOC_IRQSTATA) & (1<<6)) == 0);
		restore_flags(flags);
	}
}
 
static inline void iic_start (void)
{
	unsigned char out;
 
	out = inb(IOC_CONTROL) | FORCE_ONES | 0x02;
 
	outb(out, IOC_CONTROL);
	iic_delay();
 
	outb(out ^ 1, IOC_CONTROL);
	iic_delay();
}
 
static inline void iic_stop (void)
{
	unsigned char out;
 
	out = inb(IOC_CONTROL) | FORCE_ONES | 0x03;
 
	iic_delay();
	outb(out ^ 1, IOC_CONTROL);
 
	iic_delay();
	outb(out, IOC_CONTROL);
}
 
static int iic_sendbyte (unsigned char b)
{
	unsigned char out, in;
	int i;
 
	out = (inb(IOC_CONTROL) & 0xfc) | FORCE_ONES;
 
	outb(out, IOC_CONTROL);
	for (i = 7; i >= 0; i--) {
		unsigned char c;
		c = out | ((b & (1 << i)) ? 1 : 0);
 
		outb(c, IOC_CONTROL);
		iic_delay();
 
		outb(c | 2, IOC_CONTROL);
		iic_delay();
 
		outb(c, IOC_CONTROL);
	}
	outb(out | 1, IOC_CONTROL);
	iic_delay();
 
	outb(out | 3, IOC_CONTROL);
	iic_delay();
 
	in = inb(IOC_CONTROL) & 1;
 
	outb(out | 1, IOC_CONTROL);
	iic_delay();
 
	outb(out, IOC_CONTROL);
	iic_delay();
 
	if(in) {
		printk("No acknowledge from RTC\n");
		return 1;
	} else
		return 0;
}
 
static unsigned char iic_recvbyte (void)
{
	unsigned char out, in;
	int i;
 
	out = (inb(IOC_CONTROL) & 0xfc) | FORCE_ONES;
 
	outb(out, IOC_CONTROL);
	in = 0;
	for (i = 7; i >= 0; i--) {
		outb(out | 1, IOC_CONTROL);
		iic_delay();
		outb(out | 3, IOC_CONTROL);
		iic_delay();
		in = (in << 1) | (inb(IOC_CONTROL) & 1);
		outb(out | 1, IOC_CONTROL);
		iic_delay();
	}
	outb(out, IOC_CONTROL);
	iic_delay();
	outb(out | 2, IOC_CONTROL);
	iic_delay();
 
	return in;
}
 
void iic_control (unsigned char addr, unsigned char loc, unsigned char *buf, int len)
{
	iic_start();
 
	if (iic_sendbyte(addr & 0xfe))
		goto error;
 
	if (iic_sendbyte(loc))
		goto error;
 
	if (addr & 1) {
		int i;
 
		for (i = 0; i < len; i++)
			if (iic_sendbyte (buf[i]))
				break;
	} else {
		int i;
 
		iic_stop();
		iic_start();
		iic_sendbyte(addr|1);
		for (i = 0; i < len; i++)
			buf[i] = iic_recvbyte ();
	}
error:
	iic_stop();
}
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.