281 lines
8.2 KiB
C
281 lines
8.2 KiB
C
|
/*
|
||
|
* Copyright (c) 2009, Google Inc.
|
||
|
* All rights reserved.
|
||
|
*
|
||
|
* Copyright (c) 2009-2011, The Linux Foundation. All rights reserved.
|
||
|
*
|
||
|
* Redistribution and use in source and binary forms, with or without
|
||
|
* modification, are permitted provided that the following conditions
|
||
|
* are met:
|
||
|
* * Redistributions of source code must retain the above copyright
|
||
|
* notice, this list of conditions and the following disclaimer.
|
||
|
* * Redistributions in binary form must reproduce the above copyright
|
||
|
* notice, this list of conditions and the following disclaimer in
|
||
|
* the documentation and/or other materials provided with the
|
||
|
* distribution.
|
||
|
* * Neither the name of Google, Inc. nor the names of its contributors
|
||
|
* may be used to endorse or promote products derived from this
|
||
|
* software without specific prior written permission.
|
||
|
*
|
||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
||
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
||
|
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
||
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
||
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
||
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
|
||
|
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||
|
* SUCH DAMAGE.
|
||
|
*/
|
||
|
#include <debug.h>
|
||
|
#include <reg.h>
|
||
|
#include <sys/types.h>
|
||
|
#include <dev/ssbi.h>
|
||
|
#ifdef TARGET_USES_RSPIN_LOCK
|
||
|
#include <platform/remote_spinlock.h>
|
||
|
#endif
|
||
|
|
||
|
int i2c_ssbi_poll_for_device_ready(void)
|
||
|
{
|
||
|
unsigned long timeout = SSBI_TIMEOUT_US;
|
||
|
|
||
|
while (!(readl(MSM_SSBI_BASE + SSBI2_STATUS) & SSBI_STATUS_READY)) {
|
||
|
if (--timeout == 0) {
|
||
|
dprintf(INFO, "In Device ready function:Timeout, status %x\n", readl(MSM_SSBI_BASE + SSBI2_STATUS));
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int i2c_ssbi_poll_for_read_completed(void)
|
||
|
{
|
||
|
unsigned long timeout = SSBI_TIMEOUT_US;
|
||
|
|
||
|
while (!(readl(MSM_SSBI_BASE + SSBI2_STATUS) & SSBI_STATUS_RD_READY)) {
|
||
|
if (--timeout == 0) {
|
||
|
dprintf(INFO, "In read completed function:Timeout, status %x\n", readl(MSM_SSBI_BASE + SSBI2_STATUS));
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int i2c_ssbi_read_bytes(unsigned char *buffer, unsigned short length,
|
||
|
unsigned short slave_addr)
|
||
|
{
|
||
|
int ret = 0;
|
||
|
unsigned char *buf = buffer;
|
||
|
unsigned short len = length;
|
||
|
unsigned short addr = slave_addr;
|
||
|
unsigned long read_cmd = 0;
|
||
|
unsigned long mode2 = 0;
|
||
|
|
||
|
/*
|
||
|
* Use remote spin locks since SSBI2 controller is shared with nonHLOS proc
|
||
|
*/
|
||
|
#if TARGET_USES_RSPIN_LOCK
|
||
|
remote_spin_lock(rlock);
|
||
|
#endif
|
||
|
read_cmd = SSBI_CMD_READ(addr);
|
||
|
mode2 = readl(MSM_SSBI_BASE + SSBI2_MODE2);
|
||
|
|
||
|
//buf = alloc(len * sizeof(8));
|
||
|
if (mode2 & SSBI_MODE2_SSBI2_MODE)
|
||
|
writel(SSBI_MODE2_REG_ADDR_15_8(mode2, addr),
|
||
|
MSM_SSBI_BASE + SSBI2_MODE2);
|
||
|
|
||
|
while (len) {
|
||
|
ret = i2c_ssbi_poll_for_device_ready();
|
||
|
if (ret) {
|
||
|
dprintf (CRITICAL, "Error: device not ready\n");
|
||
|
goto end;
|
||
|
}
|
||
|
|
||
|
writel(read_cmd, MSM_SSBI_BASE + SSBI2_CMD);
|
||
|
|
||
|
ret = i2c_ssbi_poll_for_read_completed();
|
||
|
if (ret) {
|
||
|
dprintf (CRITICAL, "Error: read not completed\n");
|
||
|
goto end;
|
||
|
}
|
||
|
|
||
|
*buf++ = readl(MSM_SSBI_BASE + SSBI2_RD) & SSBI_RD_REG_DATA_MASK;
|
||
|
len--;
|
||
|
}
|
||
|
end:
|
||
|
#if TARGET_USES_RSPIN_LOCK
|
||
|
remote_spin_unlock(rlock);
|
||
|
#endif
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
int i2c_ssbi_write_bytes(unsigned char *buffer, unsigned short length,
|
||
|
unsigned short slave_addr)
|
||
|
{
|
||
|
int ret = 0;
|
||
|
unsigned long timeout = SSBI_TIMEOUT_US;
|
||
|
unsigned char *buf = buffer;
|
||
|
unsigned short len = length;
|
||
|
unsigned short addr = slave_addr;
|
||
|
unsigned long mode2 = 0;
|
||
|
|
||
|
/*
|
||
|
* Use remote spin locks since SSBI2 controller is shared with nonHLOS proc
|
||
|
*/
|
||
|
#if TARGET_USES_RSPIN_LOCK
|
||
|
remote_spin_lock(rlock);
|
||
|
#endif
|
||
|
mode2 = readl(MSM_SSBI_BASE + SSBI2_MODE2);
|
||
|
|
||
|
if (mode2 & SSBI_MODE2_SSBI2_MODE)
|
||
|
writel(SSBI_MODE2_REG_ADDR_15_8(mode2, addr),
|
||
|
MSM_SSBI_BASE + SSBI2_MODE2);
|
||
|
|
||
|
while (len) {
|
||
|
ret = i2c_ssbi_poll_for_device_ready();
|
||
|
if (ret) {
|
||
|
dprintf (CRITICAL, "Error: device not ready\n");
|
||
|
goto end;
|
||
|
}
|
||
|
|
||
|
writel(SSBI_CMD_WRITE(addr, *buf++), MSM_SSBI_BASE + SSBI2_CMD);
|
||
|
|
||
|
while (readl(MSM_SSBI_BASE + SSBI2_STATUS) & SSBI_STATUS_MCHN_BUSY) {
|
||
|
if (--timeout == 0) {
|
||
|
dprintf(INFO, "In Device ready function:Timeout, status %x\n", readl(MSM_SSBI_BASE + SSBI2_STATUS));
|
||
|
ret = 1;
|
||
|
goto end;
|
||
|
}
|
||
|
}
|
||
|
len--;
|
||
|
}
|
||
|
end:
|
||
|
#if TARGET_USES_RSPIN_LOCK
|
||
|
remote_spin_unlock(rlock);
|
||
|
#endif
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
int pa1_ssbi2_read_bytes(unsigned char *buffer, unsigned short length,
|
||
|
unsigned short slave_addr)
|
||
|
{
|
||
|
unsigned val = 0x0;
|
||
|
unsigned temp = 0x0000;
|
||
|
unsigned char *buf = buffer;
|
||
|
unsigned short len = length;
|
||
|
unsigned short addr = slave_addr;
|
||
|
unsigned long timeout = SSBI_TIMEOUT_US;
|
||
|
|
||
|
while(len)
|
||
|
{
|
||
|
val |= ((addr << PA1_SSBI2_REG_ADDR_SHIFT) |
|
||
|
(PA1_SSBI2_CMD_READ << PA1_SSBI2_CMD_RDWRN_SHIFT));
|
||
|
writel(val, PA1_SSBI2_CMD);
|
||
|
while(!((temp = readl(PA1_SSBI2_RD_STATUS)) & (1 << PA1_SSBI2_TRANS_DONE_SHIFT))) {
|
||
|
if (--timeout == 0) {
|
||
|
dprintf(INFO, "In Device ready function:Timeout\n");
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
len--;
|
||
|
*buf++ = (temp & (PA1_SSBI2_REG_DATA_MASK << PA1_SSBI2_REG_DATA_SHIFT));
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int pa1_ssbi2_write_bytes(unsigned char *buffer, unsigned short length,
|
||
|
unsigned short slave_addr)
|
||
|
{
|
||
|
unsigned val;
|
||
|
unsigned char *buf = buffer;
|
||
|
unsigned short len = length;
|
||
|
unsigned short addr = slave_addr;
|
||
|
unsigned temp = 0x00;
|
||
|
unsigned char written_data1 = 0x00;
|
||
|
unsigned long timeout = SSBI_TIMEOUT_US;
|
||
|
//unsigned char written_data2 = 0x00;
|
||
|
|
||
|
while(len)
|
||
|
{
|
||
|
temp = 0x00;
|
||
|
written_data1 = 0x00;
|
||
|
val = (addr << PA1_SSBI2_REG_ADDR_SHIFT) |
|
||
|
(PA1_SSBI2_CMD_WRITE << PA1_SSBI2_CMD_RDWRN_SHIFT) |
|
||
|
(*buf & 0xFF);
|
||
|
writel(val, PA1_SSBI2_CMD);
|
||
|
while(!((temp = readl(PA1_SSBI2_RD_STATUS)) & (1 << PA1_SSBI2_TRANS_DONE_SHIFT))) {
|
||
|
if (--timeout == 0) {
|
||
|
dprintf(INFO, "In Device write function:Timeout\n");
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
len--;
|
||
|
buf++;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int pa2_ssbi2_read_bytes(unsigned char *buffer, unsigned short length,
|
||
|
unsigned short slave_addr)
|
||
|
{
|
||
|
unsigned val = 0x0;
|
||
|
unsigned temp = 0x0000;
|
||
|
unsigned char *buf = buffer;
|
||
|
unsigned short len = length;
|
||
|
unsigned short addr = slave_addr;
|
||
|
unsigned long timeout = SSBI_TIMEOUT_US;
|
||
|
|
||
|
while(len)
|
||
|
{
|
||
|
val |= ((addr << PA2_SSBI2_REG_ADDR_SHIFT) |
|
||
|
(PA2_SSBI2_CMD_READ << PA2_SSBI2_CMD_RDWRN_SHIFT));
|
||
|
writel(val, PA2_SSBI2_CMD);
|
||
|
while(!((temp = readl(PA2_SSBI2_RD_STATUS)) & (1 << PA2_SSBI2_TRANS_DONE_SHIFT))) {
|
||
|
if (--timeout == 0) {
|
||
|
dprintf(INFO, "In Device ready function:Timeout\n");
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
len--;
|
||
|
*buf++ = (temp & (PA2_SSBI2_REG_DATA_MASK << PA2_SSBI2_REG_DATA_SHIFT));
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int pa2_ssbi2_write_bytes(unsigned char *buffer, unsigned short length,
|
||
|
unsigned short slave_addr)
|
||
|
{
|
||
|
unsigned val;
|
||
|
unsigned char *buf = buffer;
|
||
|
unsigned short len = length;
|
||
|
unsigned short addr = slave_addr;
|
||
|
unsigned temp = 0x00;
|
||
|
unsigned char written_data1 = 0x00;
|
||
|
unsigned long timeout = SSBI_TIMEOUT_US;
|
||
|
|
||
|
while(len)
|
||
|
{
|
||
|
temp = 0x00;
|
||
|
written_data1 = 0x00;
|
||
|
val = (addr << PA2_SSBI2_REG_ADDR_SHIFT) |
|
||
|
(PA2_SSBI2_CMD_WRITE << PA2_SSBI2_CMD_RDWRN_SHIFT) |
|
||
|
(*buf & 0xFF);
|
||
|
writel(val, PA2_SSBI2_CMD);
|
||
|
while(!((temp = readl(PA2_SSBI2_RD_STATUS)) & (1 << PA2_SSBI2_TRANS_DONE_SHIFT))) {
|
||
|
if (--timeout == 0) {
|
||
|
dprintf(INFO, "In Device write function:Timeout\n");
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
len--;
|
||
|
buf++;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|