mirror of
https://github.com/Telecominfraproject/OpenCellular.git
synced 2026-01-08 16:41:55 +00:00
This provides an interface to detach and re-attach fake I2C devices, which can be used to test I2C connection failure. BUG=chrome-os-partner:19235 TEST=Pass sbs_charging test along with the next CL BRANCH=None Change-Id: Ibfee79b13d45e62377d894aa28547e77bef2189e Signed-off-by: Vic Yang <victoryang@chromium.org> Reviewed-on: https://chromium-review.googlesource.com/170752 Reviewed-by: Randall Spangler <rspangler@chromium.org>
153 lines
3.4 KiB
C
153 lines
3.4 KiB
C
/* Copyright (c) 2013 The Chromium OS Authors. All rights reserved.
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
* found in the LICENSE file.
|
|
*
|
|
* Dummy I2C driver for unit test.
|
|
*/
|
|
|
|
#include "hooks.h"
|
|
#include "i2c.h"
|
|
#include "link_defs.h"
|
|
#include "test_util.h"
|
|
|
|
#define MAX_DETACHED_DEV_COUNT 3
|
|
|
|
struct i2c_dev {
|
|
int port;
|
|
int slave_addr;
|
|
int valid;
|
|
};
|
|
|
|
static struct i2c_dev detached_devs[MAX_DETACHED_DEV_COUNT];
|
|
|
|
static void detach_init(void)
|
|
{
|
|
int i;
|
|
for (i = 0; i < MAX_DETACHED_DEV_COUNT; ++i)
|
|
detached_devs[i].valid = 0;
|
|
}
|
|
DECLARE_HOOK(HOOK_INIT, detach_init, HOOK_PRIO_FIRST);
|
|
|
|
int test_detach_i2c(int port, int slave_addr)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < MAX_DETACHED_DEV_COUNT; ++i)
|
|
if (detached_devs[i].valid == 0)
|
|
break;
|
|
|
|
if (i == MAX_DETACHED_DEV_COUNT)
|
|
return EC_ERROR_OVERFLOW;
|
|
|
|
detached_devs[i].port = port;
|
|
detached_devs[i].slave_addr = slave_addr;
|
|
detached_devs[i].valid = 1;
|
|
|
|
return EC_SUCCESS;
|
|
}
|
|
|
|
int test_attach_i2c(int port, int slave_addr)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < MAX_DETACHED_DEV_COUNT; ++i)
|
|
if (detached_devs[i].valid &&
|
|
detached_devs[i].port == port &&
|
|
detached_devs[i].slave_addr == slave_addr)
|
|
break;
|
|
|
|
if (i == MAX_DETACHED_DEV_COUNT)
|
|
return EC_ERROR_INVAL;
|
|
|
|
detached_devs[i].valid = 0;
|
|
return EC_SUCCESS;
|
|
}
|
|
|
|
static int test_check_detached(int port, int slave_addr)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < MAX_DETACHED_DEV_COUNT; ++i)
|
|
if (detached_devs[i].valid &&
|
|
detached_devs[i].port == port &&
|
|
detached_devs[i].slave_addr == slave_addr)
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
int i2c_read16(int port, int slave_addr, int offset, int *data)
|
|
{
|
|
const struct test_i2c_read_dev *p;
|
|
int rv;
|
|
|
|
if (test_check_detached(port, slave_addr))
|
|
return EC_ERROR_UNKNOWN;
|
|
for (p = __test_i2c_read16; p < __test_i2c_read16_end; ++p) {
|
|
rv = p->routine(port, slave_addr, offset, data);
|
|
if (rv != EC_ERROR_INVAL)
|
|
return rv;
|
|
}
|
|
return EC_ERROR_UNKNOWN;
|
|
}
|
|
|
|
int i2c_write16(int port, int slave_addr, int offset, int data)
|
|
{
|
|
const struct test_i2c_write_dev *p;
|
|
int rv;
|
|
|
|
if (test_check_detached(port, slave_addr))
|
|
return EC_ERROR_UNKNOWN;
|
|
for (p = __test_i2c_write16; p < __test_i2c_write16_end; ++p) {
|
|
rv = p->routine(port, slave_addr, offset, data);
|
|
if (rv != EC_ERROR_INVAL)
|
|
return rv;
|
|
}
|
|
return EC_ERROR_UNKNOWN;
|
|
}
|
|
|
|
int i2c_read8(int port, int slave_addr, int offset, int *data)
|
|
{
|
|
const struct test_i2c_read_dev *p;
|
|
int rv;
|
|
|
|
if (test_check_detached(port, slave_addr))
|
|
return EC_ERROR_UNKNOWN;
|
|
for (p = __test_i2c_read8; p < __test_i2c_read8_end; ++p) {
|
|
rv = p->routine(port, slave_addr, offset, data);
|
|
if (rv != EC_ERROR_INVAL)
|
|
return rv;
|
|
}
|
|
return EC_ERROR_UNKNOWN;
|
|
}
|
|
|
|
int i2c_write8(int port, int slave_addr, int offset, int data)
|
|
{
|
|
const struct test_i2c_write_dev *p;
|
|
int rv;
|
|
|
|
if (test_check_detached(port, slave_addr))
|
|
return EC_ERROR_UNKNOWN;
|
|
for (p = __test_i2c_write8; p < __test_i2c_write8_end; ++p) {
|
|
rv = p->routine(port, slave_addr, offset, data);
|
|
if (rv != EC_ERROR_INVAL)
|
|
return rv;
|
|
}
|
|
return EC_ERROR_UNKNOWN;
|
|
}
|
|
|
|
int i2c_read_string(int port, int slave_addr, int offset, uint8_t *data,
|
|
int len)
|
|
{
|
|
const struct test_i2c_read_string_dev *p;
|
|
int rv;
|
|
|
|
if (test_check_detached(port, slave_addr))
|
|
return EC_ERROR_UNKNOWN;
|
|
for (p = __test_i2c_read_string; p < __test_i2c_read_string_end; ++p) {
|
|
rv = p->routine(port, slave_addr, offset, data, len);
|
|
if (rv != EC_ERROR_INVAL)
|
|
return rv;
|
|
}
|
|
return EC_ERROR_UNKNOWN;
|
|
}
|