277 lines
6.5 KiB
C
277 lines
6.5 KiB
C
|
/*
|
||
|
* (C) Copyright 2007-2013
|
||
|
* Allwinner Technology Co., Ltd. <www.allwinnertech.com>
|
||
|
* Jerry Wang <wangflord@allwinnertech.com>
|
||
|
*
|
||
|
* See file CREDITS for list of people who contributed to this
|
||
|
* project.
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or
|
||
|
* modify it under the terms of the GNU General Public License as
|
||
|
* published by the Free Software Foundation; either version 2 of
|
||
|
* the License, or (at your option) any later version.
|
||
|
*
|
||
|
* This program is distributed in the hope that it will be useful,
|
||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
* GNU General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with this program; if not, write to the Free Software
|
||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
|
||
|
* MA 02111-1307 USA
|
||
|
*/
|
||
|
|
||
|
#include <common.h>
|
||
|
#include <asm/io.h>
|
||
|
#include <asm/arch/p2wi.h>
|
||
|
|
||
|
static void p2wi_set_pmu_mode(u32 slave_addr, u32 reg, u32 data);
|
||
|
static void p2wi_set_clk(u32 sck);
|
||
|
static s32 p2wi_wait_state(void);
|
||
|
|
||
|
s32 p2wi_init(void)
|
||
|
{
|
||
|
uint reg_val;
|
||
|
#ifdef CONFIG_SUN6I_FPGA
|
||
|
//PG[24][25] used as p2wi SCK and SDA under FPGA platform.
|
||
|
writel(0x33, 0x01c20800 + 0xE4);
|
||
|
#endif
|
||
|
reg_val = readl(0x1f01400 + 0x28);
|
||
|
reg_val |= (1 << 0);
|
||
|
writel(reg_val, 0x1f01400 + 0x28);
|
||
|
|
||
|
reg_val = readl(0x1f02C00 + 0x00);
|
||
|
reg_val &= ~((0x07 << 0) | (0x07 << 4));
|
||
|
reg_val |= ((0x03 << 0) | (0x03 << 4));
|
||
|
writel(reg_val, 0x1f02C00 + 0x00);
|
||
|
|
||
|
//enbale twi clock, set reset as de-assert state.
|
||
|
reg_val = readl(0x1f01400 + 0x28);
|
||
|
reg_val |= (1 << 3);
|
||
|
writel(reg_val, 0x1f01400 + 0x28);
|
||
|
|
||
|
reg_val = readl(0x1f01400 + 0xb0);
|
||
|
reg_val |= (1 << 3);
|
||
|
writel(reg_val, 0x1f01400 + 0xb0);
|
||
|
|
||
|
//reset p2wi controller
|
||
|
writel(P2WI_SOFT_RST, P2WI_REG_CTRL);
|
||
|
|
||
|
//set p2wi clock
|
||
|
debug("p2wi_set_clk....\n");
|
||
|
p2wi_set_clk(P2WI_SCK);
|
||
|
|
||
|
//pmu initialize as p2wi mode
|
||
|
p2wi_set_pmu_mode(0x68, 0x3e, 0x3e);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
s32 p2wi_exit(void)
|
||
|
{
|
||
|
__u32 reg_val;
|
||
|
|
||
|
reg_val = readl(0x1f01400 + 0xb0);
|
||
|
reg_val &= ~(1 << 3);
|
||
|
writel(reg_val, 0x1f01400 + 0xb0);
|
||
|
|
||
|
reg_val = readl(0x1f01400 + 0x28);
|
||
|
reg_val &= ~(1 << 3);
|
||
|
writel(reg_val, 0x1f01400 + 0x28);
|
||
|
|
||
|
#ifdef CONFIG_SUN6I_FPGA
|
||
|
//PG[24][25] used as p2wi SCK and SDA under FPGA platform.
|
||
|
writel(0x0, 0x01c20800 + 0xE4);
|
||
|
#endif
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
s32 p2wi_read(u8 *addr, u8 *data, u32 len)
|
||
|
{
|
||
|
u32 i;
|
||
|
u32 addr0 = 0, addr1 = 0;
|
||
|
u32 data0 = 0, data1 = 0;
|
||
|
|
||
|
if((len >= PMU_TRANS_BYTE_MAX) || (len <= 0))
|
||
|
{
|
||
|
printf("p2wi read error\n");
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < len; i++)
|
||
|
{
|
||
|
//debug("read data[%d] : addr = %x\n", i, addr[i]);
|
||
|
if (i < 4)
|
||
|
{
|
||
|
//pack 8bit addr0~addr3 into 32bit addr0
|
||
|
addr0 |= addr[i] << (i * 8);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//pack 8bit addr4~addr7 into 32bit addr1
|
||
|
addr1 |= addr[i] << ((i - 4) * 8);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//write address to register
|
||
|
writel(addr0, P2WI_REG_DADDR0);
|
||
|
writel(addr1, P2WI_REG_DADDR1);
|
||
|
writel((len - 1) | (1 << 4), P2WI_REG_DLEN);
|
||
|
|
||
|
//start transmit
|
||
|
writel(readl(P2WI_REG_CTRL) | P2WI_START_TRANS, P2WI_REG_CTRL);
|
||
|
#if 0
|
||
|
debug("read config finished\n");
|
||
|
debug("P2WI_REG_DADDR0 = %x\n", readl(P2WI_REG_DADDR0));
|
||
|
debug("P2WI_REG_DADDR1 = %x\n", readl(P2WI_REG_DADDR1));
|
||
|
debug("P2WI_REG_DLEN = %x\n", readl(P2WI_REG_DLEN));
|
||
|
debug("P2WI_REG_CTRL = %x\n", readl(P2WI_REG_CTRL));
|
||
|
debug("P2WI_REG_CCR = %x\n", readl(P2WI_REG_CCR));
|
||
|
#endif
|
||
|
//wait transfer complete
|
||
|
if (p2wi_wait_state() != 0)
|
||
|
{
|
||
|
printf("p2wi read failed\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
//read out data
|
||
|
data0 = readl(P2WI_REG_DATA0);
|
||
|
data1 = readl(P2WI_REG_DATA1);
|
||
|
#if 0
|
||
|
debug("P2WI_REG_DATA0 = %x\n", data0);
|
||
|
debug("P2WI_REG_DATA1 = %x\n", data1);
|
||
|
#endif
|
||
|
for (i = 0; i < len; i++)
|
||
|
{
|
||
|
if (i < 4)
|
||
|
{
|
||
|
data[i] = (data0 >> (i * 8)) & 0xff;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
data[i] = (data1 >> ((i - 4) * 8)) & 0xff;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
s32 p2wi_write(u8 *addr, u8 *data, u32 len)
|
||
|
{
|
||
|
u32 i;
|
||
|
u32 addr0 = 0, addr1 = 0;
|
||
|
u32 data0 = 0, data1 = 0;
|
||
|
|
||
|
if((len >= PMU_TRANS_BYTE_MAX) || (len <= 0))
|
||
|
{
|
||
|
printf("p2wi write error\n");
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < len; i++)
|
||
|
{
|
||
|
// debug("write data[%d] : data = %x, addr = %x\n", i, data[i], addr[i]);
|
||
|
if (i < 4)
|
||
|
{
|
||
|
//pack 8bit data0~data3 into 32bit data0
|
||
|
//pack 8bit addr0~addr3 into 32bit addr0
|
||
|
addr0 |= addr[i] << (i * 8);
|
||
|
data0 |= data[i] << (i * 8);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//pack 8bit data4~data7 into 32bit data1
|
||
|
//pack 8bit addr4~addr7 into 32bit addr4
|
||
|
addr1 |= addr[i] << ((i - 4) * 8);
|
||
|
data1 |= data[i] << ((i - 4) * 8);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//write register
|
||
|
writel(addr0, P2WI_REG_DADDR0);
|
||
|
writel(addr1, P2WI_REG_DADDR1);
|
||
|
writel(data0, P2WI_REG_DATA0);
|
||
|
writel(data1, P2WI_REG_DATA1);
|
||
|
writel(len - 1, P2WI_REG_DLEN);
|
||
|
|
||
|
//start transfer
|
||
|
writel(readl(P2WI_REG_CTRL) | P2WI_START_TRANS, P2WI_REG_CTRL);
|
||
|
#if 0
|
||
|
debug("write config finished\n");
|
||
|
debug("P2WI_REG_DADDR0 = %x\n", readl(P2WI_REG_DADDR0));
|
||
|
debug("P2WI_REG_DADDR1 = %x\n", readl(P2WI_REG_DADDR1));
|
||
|
debug("P2WI_REG_DATA0 = %x\n", readl(P2WI_REG_DATA0));
|
||
|
debug("P2WI_REG_DATA1 = %x\n", readl(P2WI_REG_DATA1));
|
||
|
debug("P2WI_REG_DLEN = %x\n", readl(P2WI_REG_DLEN));
|
||
|
debug("P2WI_REG_CTRL = %x\n", readl(P2WI_REG_CTRL));
|
||
|
#endif
|
||
|
//wait transfer complete
|
||
|
if (p2wi_wait_state() != 0)
|
||
|
{
|
||
|
printf("p2wi write failed\n");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static void p2wi_set_pmu_mode(u32 slave_addr, u32 reg, u32 data)
|
||
|
{
|
||
|
//set pmu work mode
|
||
|
writel(P2WI_PMU_INIT | (slave_addr) | (reg << 8) | (data << 16), P2WI_REG_PMCR);
|
||
|
|
||
|
//wait pmu mode set complete
|
||
|
while(readl(P2WI_REG_PMCR) & P2WI_PMU_INIT)
|
||
|
{
|
||
|
;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void p2wi_set_clk(u32 sck)
|
||
|
{
|
||
|
#ifndef CONFIG_SUN6I_FPGA
|
||
|
u32 src_clk = 12000000;
|
||
|
u32 div = src_clk / sck / 2 - 1;
|
||
|
u32 sda_odly = div >> 1;
|
||
|
u32 rval = div | (sda_odly << 8);
|
||
|
writel(rval, P2WI_REG_CCR);
|
||
|
#else
|
||
|
//fpga platform: apb clock = 24M, p2wi clock = 3M.
|
||
|
writel(0x103, P2WI_REG_CCR);
|
||
|
debug("P2WI_REG_CCR value = %x\n", readl(P2WI_REG_CCR));
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
static s32 p2wi_wait_state(void)
|
||
|
{
|
||
|
s32 ret = -1;
|
||
|
u32 stat;
|
||
|
|
||
|
while (1)
|
||
|
{
|
||
|
stat = readl(P2WI_REG_STAT);
|
||
|
if (stat & P2WI_TERR_INT)
|
||
|
{
|
||
|
//transfer error
|
||
|
printf("p2wi transfer error [%x]\n", ((stat >> 8) & 0xff));
|
||
|
ret = -1;
|
||
|
break;
|
||
|
}
|
||
|
if (stat & P2WI_TOVER_INT)
|
||
|
{
|
||
|
//transfer complete
|
||
|
ret = 0;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
//clear state flag
|
||
|
writel(stat, P2WI_REG_STAT);
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|