| // SPDX-License-Identifier: GPL-2.0+ |
| /* |
| * Copyright (C) 2015 Google, Inc |
| */ |
| |
| #include <common.h> |
| #include <dm.h> |
| #include <log.h> |
| #include <mapmem.h> |
| #include <regmap.h> |
| #include <syscon.h> |
| #include <rand.h> |
| #include <asm/test.h> |
| #include <dm/test.h> |
| #include <dm/devres.h> |
| #include <linux/err.h> |
| #include <test/test.h> |
| #include <test/ut.h> |
| |
| /* Base test of register maps */ |
| static int dm_test_regmap_base(struct unit_test_state *uts) |
| { |
| struct udevice *dev; |
| struct regmap *map; |
| ofnode node; |
| int i; |
| |
| ut_assertok(uclass_get_device(UCLASS_SYSCON, 0, &dev)); |
| map = syscon_get_regmap(dev); |
| ut_assertok_ptr(map); |
| ut_asserteq(1, map->range_count); |
| ut_asserteq(0x10, map->ranges[0].start); |
| ut_asserteq(16, map->ranges[0].size); |
| ut_asserteq(0x10, map_to_sysmem(regmap_get_range(map, 0))); |
| |
| ut_assertok(uclass_get_device(UCLASS_SYSCON, 1, &dev)); |
| map = syscon_get_regmap(dev); |
| ut_assertok_ptr(map); |
| ut_asserteq(4, map->range_count); |
| ut_asserteq(0x20, map->ranges[0].start); |
| for (i = 0; i < 4; i++) { |
| const unsigned long addr = 0x20 + 8 * i; |
| |
| ut_asserteq(addr, map->ranges[i].start); |
| ut_asserteq(5 + i, map->ranges[i].size); |
| ut_asserteq(addr, map_to_sysmem(regmap_get_range(map, i))); |
| } |
| |
| /* Check that we can't pretend a different device is a syscon */ |
| ut_assertok(uclass_get_device(UCLASS_I2C, 0, &dev)); |
| map = syscon_get_regmap(dev); |
| ut_asserteq_ptr(ERR_PTR(-ENOEXEC), map); |
| |
| /* A different device can be a syscon by using Linux-compat API */ |
| node = ofnode_path("/syscon@2"); |
| ut_assert(ofnode_valid(node)); |
| |
| map = syscon_node_to_regmap(node); |
| ut_assertok_ptr(map); |
| ut_asserteq(4, map->range_count); |
| ut_asserteq(0x40, map->ranges[0].start); |
| for (i = 0; i < 4; i++) { |
| const unsigned long addr = 0x40 + 8 * i; |
| |
| ut_asserteq(addr, map->ranges[i].start); |
| ut_asserteq(5 + i, map->ranges[i].size); |
| ut_asserteq(addr, map_to_sysmem(regmap_get_range(map, i))); |
| } |
| |
| return 0; |
| } |
| DM_TEST(dm_test_regmap_base, UT_TESTF_SCAN_PDATA | UT_TESTF_SCAN_FDT); |
| |
| /* Test we can access a regmap through syscon */ |
| static int dm_test_regmap_syscon(struct unit_test_state *uts) |
| { |
| struct regmap *map; |
| |
| map = syscon_get_regmap_by_driver_data(SYSCON0); |
| ut_assertok_ptr(map); |
| ut_asserteq(1, map->range_count); |
| |
| map = syscon_get_regmap_by_driver_data(SYSCON1); |
| ut_assertok_ptr(map); |
| ut_asserteq(4, map->range_count); |
| |
| map = syscon_get_regmap_by_driver_data(SYSCON_COUNT); |
| ut_asserteq_ptr(ERR_PTR(-ENODEV), map); |
| |
| ut_asserteq(0x10, map_to_sysmem(syscon_get_first_range(SYSCON0))); |
| ut_asserteq(0x20, map_to_sysmem(syscon_get_first_range(SYSCON1))); |
| ut_asserteq_ptr(ERR_PTR(-ENODEV), |
| syscon_get_first_range(SYSCON_COUNT)); |
| |
| return 0; |
| } |
| |
| DM_TEST(dm_test_regmap_syscon, UT_TESTF_SCAN_PDATA | UT_TESTF_SCAN_FDT); |
| |
| /* Read/Write/Modify test */ |
| static int dm_test_regmap_rw(struct unit_test_state *uts) |
| { |
| struct udevice *dev; |
| struct regmap *map; |
| uint reg; |
| |
| sandbox_set_enable_memio(true); |
| ut_assertok(uclass_get_device(UCLASS_SYSCON, 0, &dev)); |
| map = syscon_get_regmap(dev); |
| ut_assertok_ptr(map); |
| |
| ut_assertok(regmap_write(map, 0, 0xcacafafa)); |
| ut_assertok(regmap_write(map, 5, 0x55aa2211)); |
| |
| ut_assertok(regmap_read(map, 0, ®)); |
| ut_asserteq(0xcacafafa, reg); |
| ut_assertok(regmap_read(map, 5, ®)); |
| ut_asserteq(0x55aa2211, reg); |
| |
| ut_assertok(regmap_read(map, 0, ®)); |
| ut_asserteq(0xcacafafa, reg); |
| ut_assertok(regmap_update_bits(map, 0, 0xff00ff00, 0x55aa2211)); |
| ut_assertok(regmap_read(map, 0, ®)); |
| ut_asserteq(0x55ca22fa, reg); |
| ut_assertok(regmap_update_bits(map, 5, 0x00ff00ff, 0xcacafada)); |
| ut_assertok(regmap_read(map, 5, ®)); |
| ut_asserteq(0x55ca22da, reg); |
| |
| return 0; |
| } |
| |
| DM_TEST(dm_test_regmap_rw, UT_TESTF_SCAN_PDATA | UT_TESTF_SCAN_FDT); |
| |
| /* Get/Set test */ |
| static int dm_test_regmap_getset(struct unit_test_state *uts) |
| { |
| struct udevice *dev; |
| struct regmap *map; |
| uint reg; |
| struct layout { |
| u32 val0; |
| u32 val1; |
| u32 val2; |
| u32 val3; |
| }; |
| |
| sandbox_set_enable_memio(true); |
| ut_assertok(uclass_get_device(UCLASS_SYSCON, 0, &dev)); |
| map = syscon_get_regmap(dev); |
| ut_assertok_ptr(map); |
| |
| regmap_set(map, struct layout, val0, 0xcacafafa); |
| regmap_set(map, struct layout, val3, 0x55aa2211); |
| |
| ut_assertok(regmap_get(map, struct layout, val0, ®)); |
| ut_asserteq(0xcacafafa, reg); |
| ut_assertok(regmap_get(map, struct layout, val3, ®)); |
| ut_asserteq(0x55aa2211, reg); |
| |
| return 0; |
| } |
| |
| DM_TEST(dm_test_regmap_getset, UT_TESTF_SCAN_PDATA | UT_TESTF_SCAN_FDT); |
| |
| /* Read polling test */ |
| static int dm_test_regmap_poll(struct unit_test_state *uts) |
| { |
| struct udevice *dev; |
| struct regmap *map; |
| uint reg; |
| unsigned long start; |
| |
| ut_assertok(uclass_get_device(UCLASS_SYSCON, 0, &dev)); |
| map = syscon_get_regmap(dev); |
| ut_assertok_ptr(map); |
| |
| start = get_timer(0); |
| |
| ut_assertok(regmap_write(map, 0, 0x0)); |
| ut_asserteq(-ETIMEDOUT, |
| regmap_read_poll_timeout_test(map, 0, reg, |
| (reg == 0xcacafafa), |
| 1, 5 * CONFIG_SYS_HZ, |
| 5 * CONFIG_SYS_HZ)); |
| |
| ut_assert(get_timer(start) > (5 * CONFIG_SYS_HZ)); |
| |
| return 0; |
| } |
| |
| DM_TEST(dm_test_regmap_poll, UT_TESTF_SCAN_PDATA | UT_TESTF_SCAN_FDT); |
| |
| struct regmaptest_priv { |
| struct regmap *cfg_regmap; /* For testing regmap_config options. */ |
| struct regmap *fld_regmap; /* For testing regmap fields. */ |
| struct regmap_field **fields; |
| }; |
| |
| static const struct reg_field field_cfgs[] = { |
| { |
| .reg = 0, |
| .lsb = 0, |
| .msb = 6, |
| }, |
| { |
| .reg = 2, |
| .lsb = 4, |
| .msb = 12, |
| }, |
| { |
| .reg = 2, |
| .lsb = 12, |
| .msb = 15, |
| } |
| }; |
| |
| #define REGMAP_TEST_BUF_START 0 |
| #define REGMAP_TEST_BUF_SZ 5 |
| |
| static int remaptest_probe(struct udevice *dev) |
| { |
| struct regmaptest_priv *priv = dev_get_priv(dev); |
| struct regmap *regmap; |
| struct regmap_field *field; |
| struct regmap_config cfg; |
| int i; |
| static const int n = ARRAY_SIZE(field_cfgs); |
| |
| /* |
| * To exercise all the regmap config options, create a regmap that |
| * points to a custom memory area instead of the one defined in device |
| * tree. Use 2-byte elements. To allow directly indexing into the |
| * elements, use an offset shift of 1. So, accessing offset 1 gets the |
| * element at index 1 at memory location 2. |
| * |
| * REGMAP_TEST_BUF_SZ is the number of elements, so we need to multiply |
| * it by 2 because r_size expects number of bytes. |
| */ |
| cfg.reg_offset_shift = 1; |
| cfg.r_start = REGMAP_TEST_BUF_START; |
| cfg.r_size = REGMAP_TEST_BUF_SZ * 2; |
| cfg.width = REGMAP_SIZE_16; |
| |
| regmap = devm_regmap_init(dev, NULL, NULL, &cfg); |
| if (IS_ERR(regmap)) |
| return PTR_ERR(regmap); |
| priv->cfg_regmap = regmap; |
| |
| memset(&cfg, 0, sizeof(struct regmap_config)); |
| cfg.width = REGMAP_SIZE_16; |
| |
| regmap = devm_regmap_init(dev, NULL, NULL, &cfg); |
| if (IS_ERR(regmap)) |
| return PTR_ERR(regmap); |
| priv->fld_regmap = regmap; |
| |
| priv->fields = devm_kzalloc(dev, sizeof(struct regmap_field *) * n, |
| GFP_KERNEL); |
| if (!priv->fields) |
| return -ENOMEM; |
| |
| for (i = 0 ; i < n; i++) { |
| field = devm_regmap_field_alloc(dev, priv->fld_regmap, |
| field_cfgs[i]); |
| if (IS_ERR(field)) |
| return PTR_ERR(field); |
| priv->fields[i] = field; |
| } |
| |
| return 0; |
| } |
| |
| static const struct udevice_id regmaptest_ids[] = { |
| { .compatible = "sandbox,regmap_test" }, |
| { } |
| }; |
| |
| U_BOOT_DRIVER(regmap_test) = { |
| .name = "regmaptest_drv", |
| .of_match = regmaptest_ids, |
| .id = UCLASS_NOP, |
| .probe = remaptest_probe, |
| .priv_auto = sizeof(struct regmaptest_priv), |
| }; |
| |
| static int dm_test_devm_regmap(struct unit_test_state *uts) |
| { |
| int i = 0; |
| u16 val; |
| void *valp = &val; |
| u16 pattern[REGMAP_TEST_BUF_SZ]; |
| u16 *buffer; |
| struct udevice *dev; |
| struct regmaptest_priv *priv; |
| |
| sandbox_set_enable_memio(true); |
| |
| /* |
| * Map the memory area the regmap should point to so we can make sure |
| * the writes actually go to that location. |
| */ |
| buffer = map_physmem(REGMAP_TEST_BUF_START, |
| REGMAP_TEST_BUF_SZ * 2, MAP_NOCACHE); |
| |
| ut_assertok(uclass_get_device_by_name(UCLASS_NOP, "regmap-test_0", |
| &dev)); |
| priv = dev_get_priv(dev); |
| |
| for (i = 0; i < REGMAP_TEST_BUF_SZ; i++) { |
| pattern[i] = i * 0x87654321; |
| ut_assertok(regmap_write(priv->cfg_regmap, i, pattern[i])); |
| } |
| for (i = 0; i < REGMAP_TEST_BUF_SZ; i++) { |
| ut_assertok(regmap_read(priv->cfg_regmap, i, valp)); |
| ut_asserteq(val, buffer[i]); |
| ut_asserteq(val, pattern[i]); |
| } |
| |
| ut_asserteq(-ERANGE, regmap_write(priv->cfg_regmap, REGMAP_TEST_BUF_SZ, |
| val)); |
| ut_asserteq(-ERANGE, regmap_read(priv->cfg_regmap, REGMAP_TEST_BUF_SZ, |
| valp)); |
| ut_asserteq(-ERANGE, regmap_write(priv->cfg_regmap, -1, val)); |
| ut_asserteq(-ERANGE, regmap_read(priv->cfg_regmap, -1, valp)); |
| |
| return 0; |
| } |
| DM_TEST(dm_test_devm_regmap, UT_TESTF_SCAN_PDATA | UT_TESTF_SCAN_FDT); |
| |
| static int test_one_field(struct unit_test_state *uts, |
| struct regmap *regmap, |
| struct regmap_field *field, |
| struct reg_field field_cfg) |
| { |
| int j; |
| unsigned int val; |
| int mask = (1 << (field_cfg.msb - field_cfg.lsb + 1)) - 1; |
| int shift = field_cfg.lsb; |
| |
| ut_assertok(regmap_write(regmap, field_cfg.reg, 0)); |
| ut_assertok(regmap_read(regmap, field_cfg.reg, &val)); |
| ut_asserteq(0, val); |
| |
| for (j = 0; j <= mask; j++) { |
| ut_assertok(regmap_field_write(field, j)); |
| ut_assertok(regmap_field_read(field, &val)); |
| ut_asserteq(j, val); |
| ut_assertok(regmap_read(regmap, field_cfg.reg, &val)); |
| ut_asserteq(j << shift, val); |
| } |
| |
| ut_assertok(regmap_field_write(field, mask + 1)); |
| ut_assertok(regmap_read(regmap, field_cfg.reg, &val)); |
| ut_asserteq(0, val); |
| |
| ut_assertok(regmap_field_write(field, 0xFFFF)); |
| ut_assertok(regmap_read(regmap, field_cfg.reg, &val)); |
| ut_asserteq(mask << shift, val); |
| |
| ut_assertok(regmap_write(regmap, field_cfg.reg, 0xFFFF)); |
| ut_assertok(regmap_field_write(field, 0)); |
| ut_assertok(regmap_read(regmap, field_cfg.reg, &val)); |
| ut_asserteq(0xFFFF & ~(mask << shift), val); |
| return 0; |
| } |
| |
| static int dm_test_devm_regmap_field(struct unit_test_state *uts) |
| { |
| int i, rc; |
| struct udevice *dev; |
| struct regmaptest_priv *priv; |
| |
| ut_assertok(uclass_get_device_by_name(UCLASS_NOP, "regmap-test_0", |
| &dev)); |
| priv = dev_get_priv(dev); |
| |
| sandbox_set_enable_memio(true); |
| for (i = 0 ; i < ARRAY_SIZE(field_cfgs); i++) { |
| rc = test_one_field(uts, priv->fld_regmap, priv->fields[i], |
| field_cfgs[i]); |
| if (rc) |
| break; |
| } |
| |
| return 0; |
| } |
| DM_TEST(dm_test_devm_regmap_field, UT_TESTF_SCAN_PDATA | UT_TESTF_SCAN_FDT); |