// SPDX-License-Identifier: GPL-2.0+ /* * Copyright (C) 2015 Google, Inc */ #include #include #include #include #include #include #include #include #include #include #include #include #include /* 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; uint 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, &val)); 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, &val)); ut_asserteq(-ERANGE, regmap_write(priv->cfg_regmap, -1, val)); ut_asserteq(-ERANGE, regmap_read(priv->cfg_regmap, -1, &val)); 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);