This module tests Direct Memory Access to some device on LocalPlus Bus for Freescale MPC512x. In other words it tests the bundle of mpc512x_lpbfifo and mpc512x_dma drivers.
This testing driver was multiply used with static RAM (CY62167EV30LL-45ZXI) which lives on LocalPlus Bus on our board. This testing driver was used instead of the original static RAM driver and it is an abnormal hack. That is why I just provide the driver code and don't modify any environment. Signed-off-by: Alexander Popov <a13xp0p0...@gmail.com> --- drivers/misc/mpc512x_lpbdma_test.c | 310 +++++++++++++++++++++++++++++++++++++ 1 file changed, 310 insertions(+) create mode 100644 drivers/misc/mpc512x_lpbdma_test.c diff --git a/drivers/misc/mpc512x_lpbdma_test.c b/drivers/misc/mpc512x_lpbdma_test.c new file mode 100644 index 0000000..4fdd052 --- /dev/null +++ b/drivers/misc/mpc512x_lpbdma_test.c @@ -0,0 +1,310 @@ +/* + * Tests for DMA via LocalPlus Bus for the Freescale MPC512x. + * + * Copyright (C) Promcontroller, 2013. + * Author is Alexander Popov <a13xp0p0...@gmail.com>. + * + * This module tests Direct Memory Access to the devices on LocalPlus Bus. + * + * This file is released under the GPLv2. + */ + +#include <linux/kernel.h> +#include <linux/module.h> +#include <linux/of.h> +#include <linux/of_platform.h> +#include <linux/io.h> +#include <linux/workqueue.h> +#include <linux/slab.h> +#include <linux/random.h> +#include <linux/string.h> +#include <asm/mpc5121.h> + +MODULE_AUTHOR("Alexander Popov <a13xp0p0...@gmail.com>"); +MODULE_LICENSE("GPL"); + +#define DRV_NAME "mpc512x_lpbdma_test" +#define ENOUGH 131072 + +static int repeat = 1; + +enum test_iteration_result { + NEXT_TEST_PLEASE, + TRANSFER_LAUNCHED, + ALL_TESTS_DONE, +}; + +static struct lpbdma_test_data { + struct device *dev; + struct resource r_io; + void __iomem *bus_vaddr; + void *ram_vaddr; + + /* Current data */ + int current_repeat; + int current_test_case; + enum mpc512x_lpbfifo_req_dir current_dir; + struct mpc512x_lpbfifo_request current_req; +} tt; + +static void lpbdma_prepare_next_iteration(void); +static void lpbdma_testing(struct work_struct *w); +static void lpbdma_test_callback(struct mpc512x_lpbfifo_request *req); +static enum test_iteration_result lpbdma_test_iteration(unsigned int test_case, + enum mpc512x_lpbfifo_req_dir dir); + +static DECLARE_WORK(lpbdma_work, lpbdma_testing); +static struct workqueue_struct *wq; + +static void lpbdma_prepare_next_iteration(void) +{ + tt.current_repeat++; + + if (tt.current_repeat > repeat) { + tt.current_repeat = 1; + if (tt.current_dir == MPC512X_LPBFIFO_REQ_DIR_READ) + tt.current_dir = MPC512X_LPBFIFO_REQ_DIR_WRITE; + else { + tt.current_dir = MPC512X_LPBFIFO_REQ_DIR_READ; + tt.current_test_case += 1; + } + } +} + +static void lpbdma_testing(struct work_struct *w) +{ + lpbdma_prepare_next_iteration(); + while (lpbdma_test_iteration(tt.current_test_case, + tt.current_dir) == NEXT_TEST_PLEASE) { + lpbdma_prepare_next_iteration(); + } +} + +static enum test_iteration_result lpbdma_test_iteration(unsigned int test_case, + enum mpc512x_lpbfifo_req_dir dir) +{ + struct mpc512x_lpbfifo_request *req = &(tt.current_req); + int will_work; + unsigned int i; + unsigned int bufn; + u16 buf; + + /* Prepare request for data transfer */ + req->cs = 0; + req->bus_phys = tt.r_io.start; + req->ram_virt = tt.ram_vaddr; + req->dir = dir; + req->callback = lpbdma_test_callback; + + switch (test_case) { + case 0: + /* normal LPB_DEV <-> RAM transfer */ + will_work = 1; + req->size = 32768; /* 32 kBytes, for example */ + req->portsize = LPB_DEV_PORTSIZE_UNDEFINED; + break; + case 1: + /* maximum size transfer */ + will_work = 1; + req->size = 131068; /* 128 kBytes - 4 Bytes */ + req->portsize = LPB_DEV_PORTSIZE_UNDEFINED; + break; + case 2: + /* maximum transfer size is exceeded */ + will_work = 0; + req->size = 133120; /* 130 kBytes, for example */ + req->portsize = LPB_DEV_PORTSIZE_UNDEFINED; + break; + case 3: + /* LPB_DEV has own FIFO register + * It's width is 8 bytes, for example */ + will_work = 1; + req->size = 16384; /* 16 kBytes, for example */ + req->portsize = LPB_DEV_PORTSIZE_8_BYTES; + break; + case 4: + /* Ditto. But size is not aligned on portsize */ + will_work = 0; + req->size = 16382; + req->portsize = LPB_DEV_PORTSIZE_8_BYTES; + break; + case 5: + /* size is not aligned on 4 */ + will_work = 0; + req->size = 42; /* eh! */ + req->portsize = LPB_DEV_PORTSIZE_UNDEFINED; + break; + case 6: + /* ram address is not aligned on 4 + * but is aligned on 2 */ + will_work = 1; + req->ram_virt += 2; + req->size = 65536; /* 64 kBytes, for example */ + req->portsize = LPB_DEV_PORTSIZE_UNDEFINED; + break; + case 7: + /* ram address is not aligned even on 2 */ + will_work = 1; + req->ram_virt += 3; + req->size = 8192; /* 8 kBytes, for example */ + req->portsize = LPB_DEV_PORTSIZE_UNDEFINED; + break; + default: + dev_info(tt.dev, "All tests are done\n"); + return ALL_TESTS_DONE; + } + + if (will_work) { + /* prepare source data */ + if (dir == MPC512X_LPBFIFO_REQ_DIR_READ) { + if (req->portsize == LPB_DEV_PORTSIZE_UNDEFINED) + bufn = req->size / sizeof(buf); + else + bufn = req->portsize / sizeof(buf); + + for (i = 0; i < bufn; i++) { + get_random_bytes(&buf, sizeof(buf)); + out_be16(tt.bus_vaddr + i * sizeof(buf), buf); + } + } else + get_random_bytes(req->ram_virt, req->size); + } + + if (mpc512x_lpbfifo_submit(req)) { + /* submit failed */ + if (!will_work) + dev_info(tt.dev, "Test %u.%i passed\n", test_case, dir); + else + dev_err(tt.dev, "TEST %u.%i FAILED\n", test_case, dir); + return NEXT_TEST_PLEASE; + } else { + /* Request is submitted. Let's wait + * for lpbdma_test_callback() to be called */ + dev_info(tt.dev, "Test %u.%i is launched\n", test_case, dir); + return TRANSFER_LAUNCHED; + } +} + +static void lpbdma_test_callback(struct mpc512x_lpbfifo_request *req) +{ + int test_passed = 1; + int i = 0; + u16 val1, val2; + + if (req->portsize != LPB_DEV_PORTSIZE_UNDEFINED && + req->dir == MPC512X_LPBFIFO_REQ_DIR_WRITE) { + /* During that transfer the next portion of data + * overwrites the previous one. + * Let's check the last portion of data */ + + for (i = 0; i < req->portsize / sizeof(val1); i++) { + val1 = in_be16(tt.bus_vaddr + i * sizeof(val1)); + /* Since req->ram_virt + * might not be aligned: */ + memcpy(&val2, req->ram_virt + req->size - + req->portsize + i * sizeof(val2), + sizeof(val2)); + if (val1 != val2) { + test_passed = 0; + dev_err(tt.dev, "ERROR: %i: %x != %x\n", + i, val2, val1); + } + } + } else { + for (i = 0; i < req->size / sizeof(val1); i++) { + if (req->portsize != LPB_DEV_PORTSIZE_UNDEFINED) { + val1 = in_be16(tt.bus_vaddr + + i * sizeof(val1) % req->portsize); + } else + val1 = in_be16(tt.bus_vaddr + i * sizeof(val1)); + + /* Ditto */ + memcpy(&val2, req->ram_virt + i * sizeof(val2), + sizeof(val2)); + + if (val1 != val2) { + test_passed = 0; + dev_err(tt.dev, "ERROR: %i: %x != %x\n", + i, val2, val1); + } + } + } + + if (test_passed) + dev_info(tt.dev, "...Passed\n"); + else + dev_err(tt.dev, "...FAILED\n"); + + /* Next test please */ + if (wq) + queue_work(wq, &lpbdma_work); +} + +static int mpc512x_lpbdma_test_probe(struct platform_device *pdev) +{ + struct resource *rio_ptr = &tt.r_io; + tt.dev = &pdev->dev; + + if (of_address_to_resource(pdev->dev.of_node, 0, rio_ptr)) { + dev_err(tt.dev, "can't find the resource\n"); + return 1; + } + dev_info(tt.dev, "Resource on LPB: 0x%x, size %u bytes\n", + rio_ptr->start, resource_size(rio_ptr)); + + if (!request_mem_region(rio_ptr->start, + resource_size(rio_ptr), DRV_NAME)) { + dev_err(tt.dev, "can't request_mem_region\n"); + return 1; + } + + tt.bus_vaddr = ioremap(rio_ptr->start, resource_size(rio_ptr)); + if (!tt.bus_vaddr) { + dev_err(tt.dev, "ioremap failed\n"); + release_mem_region(rio_ptr->start, resource_size(rio_ptr)); + return 1; + } + + tt.ram_vaddr = kmalloc(ENOUGH, GFP_DMA | GFP_KERNEL); + + dev_info(tt.dev, "Let's go!\n"); + + tt.current_repeat = 0; + tt.current_test_case = 0; + tt.current_dir = MPC512X_LPBFIFO_REQ_DIR_READ; + + wq = create_singlethread_workqueue("mpc512x_lpbdma_test"); + if (wq) + queue_work(wq, &lpbdma_work); + + return 0; +} + +static int mpc512x_lpbdma_test_remove(struct platform_device *pdev) +{ + if (wq) + destroy_workqueue(wq); + + release_mem_region(tt.r_io.start, resource_size(&tt.r_io)); + iounmap(tt.bus_vaddr); + kfree(tt.ram_vaddr); + return 0; +} + +static const struct of_device_id lpbdma_test_match[] = { + { .compatible = "tecon,mpc512x_lpbdma_test", }, + {}, +}; + +static struct platform_driver mpc512x_lpbdma_test_driver = { + .probe = mpc512x_lpbdma_test_probe, + .remove = mpc512x_lpbdma_test_remove, + .driver = { + .name = DRV_NAME, + .owner = THIS_MODULE, + .of_match_table = lpbdma_test_match, + }, +}; +module_platform_driver(mpc512x_lpbdma_test_driver); + +module_param(repeat, int, 0); -- 1.7.11.3 _______________________________________________ Linuxppc-dev mailing list Linuxppc-dev@lists.ozlabs.org https://lists.ozlabs.org/listinfo/linuxppc-dev