dw-xdata-pcie.c 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (c) 2020 Synopsys, Inc. and/or its affiliates.
  4. * Synopsys DesignWare xData driver
  5. *
  6. * Author: Gustavo Pimentel <gustavo.pimentel@synopsys.com>
  7. */
  8. #include <linux/miscdevice.h>
  9. #include <linux/bitfield.h>
  10. #include <linux/pci-epf.h>
  11. #include <linux/kernel.h>
  12. #include <linux/module.h>
  13. #include <linux/device.h>
  14. #include <linux/bitops.h>
  15. #include <linux/mutex.h>
  16. #include <linux/delay.h>
  17. #include <linux/pci.h>
  18. #include <linux/string_choices.h>
  19. #define DW_XDATA_DRIVER_NAME "dw-xdata-pcie"
  20. #define DW_XDATA_EP_MEM_OFFSET 0x8000000
  21. static DEFINE_IDA(xdata_ida);
  22. #define STATUS_DONE BIT(0)
  23. #define CONTROL_DOORBELL BIT(0)
  24. #define CONTROL_IS_WRITE BIT(1)
  25. #define CONTROL_LENGTH(a) FIELD_PREP(GENMASK(13, 2), a)
  26. #define CONTROL_PATTERN_INC BIT(16)
  27. #define CONTROL_NO_ADDR_INC BIT(18)
  28. #define XPERF_CONTROL_ENABLE BIT(5)
  29. #define BURST_REPEAT BIT(31)
  30. #define BURST_VALUE 0x1001
  31. #define PATTERN_VALUE 0x0
  32. struct dw_xdata_regs {
  33. u32 addr_lsb; /* 0x000 */
  34. u32 addr_msb; /* 0x004 */
  35. u32 burst_cnt; /* 0x008 */
  36. u32 control; /* 0x00c */
  37. u32 pattern; /* 0x010 */
  38. u32 status; /* 0x014 */
  39. u32 RAM_addr; /* 0x018 */
  40. u32 RAM_port; /* 0x01c */
  41. u32 _reserved0[14]; /* 0x020..0x054 */
  42. u32 perf_control; /* 0x058 */
  43. u32 _reserved1[41]; /* 0x05c..0x0fc */
  44. u32 wr_cnt_lsb; /* 0x100 */
  45. u32 wr_cnt_msb; /* 0x104 */
  46. u32 rd_cnt_lsb; /* 0x108 */
  47. u32 rd_cnt_msb; /* 0x10c */
  48. } __packed;
  49. struct dw_xdata_region {
  50. phys_addr_t paddr; /* physical address */
  51. void __iomem *vaddr; /* virtual address */
  52. };
  53. struct dw_xdata {
  54. struct dw_xdata_region rg_region; /* registers */
  55. size_t max_wr_len; /* max wr xfer len */
  56. size_t max_rd_len; /* max rd xfer len */
  57. struct mutex mutex;
  58. struct pci_dev *pdev;
  59. struct miscdevice misc_dev;
  60. };
  61. static inline struct dw_xdata_regs __iomem *__dw_regs(struct dw_xdata *dw)
  62. {
  63. return dw->rg_region.vaddr;
  64. }
  65. static void dw_xdata_stop(struct dw_xdata *dw)
  66. {
  67. u32 burst;
  68. mutex_lock(&dw->mutex);
  69. burst = readl(&(__dw_regs(dw)->burst_cnt));
  70. if (burst & BURST_REPEAT) {
  71. burst &= ~(u32)BURST_REPEAT;
  72. writel(burst, &(__dw_regs(dw)->burst_cnt));
  73. }
  74. mutex_unlock(&dw->mutex);
  75. }
  76. static void dw_xdata_start(struct dw_xdata *dw, bool write)
  77. {
  78. struct device *dev = &dw->pdev->dev;
  79. u32 control, status;
  80. /* Stop first if xfer in progress */
  81. dw_xdata_stop(dw);
  82. mutex_lock(&dw->mutex);
  83. /* Clear status register */
  84. writel(0x0, &(__dw_regs(dw)->status));
  85. /* Burst count register set for continuous until stopped */
  86. writel(BURST_REPEAT | BURST_VALUE, &(__dw_regs(dw)->burst_cnt));
  87. /* Pattern register */
  88. writel(PATTERN_VALUE, &(__dw_regs(dw)->pattern));
  89. /* Control register */
  90. control = CONTROL_DOORBELL | CONTROL_PATTERN_INC | CONTROL_NO_ADDR_INC;
  91. if (write) {
  92. control |= CONTROL_IS_WRITE;
  93. control |= CONTROL_LENGTH(dw->max_wr_len);
  94. } else {
  95. control |= CONTROL_LENGTH(dw->max_rd_len);
  96. }
  97. writel(control, &(__dw_regs(dw)->control));
  98. /*
  99. * The xData HW block needs about 100 ms to initiate the traffic
  100. * generation according this HW block datasheet.
  101. */
  102. usleep_range(100, 150);
  103. status = readl(&(__dw_regs(dw)->status));
  104. mutex_unlock(&dw->mutex);
  105. if (!(status & STATUS_DONE))
  106. dev_dbg(dev, "xData: started %s direction\n",
  107. str_write_read(write));
  108. }
  109. static void dw_xdata_perf_meas(struct dw_xdata *dw, u64 *data, bool write)
  110. {
  111. if (write) {
  112. *data = readl(&(__dw_regs(dw)->wr_cnt_msb));
  113. *data <<= 32;
  114. *data |= readl(&(__dw_regs(dw)->wr_cnt_lsb));
  115. } else {
  116. *data = readl(&(__dw_regs(dw)->rd_cnt_msb));
  117. *data <<= 32;
  118. *data |= readl(&(__dw_regs(dw)->rd_cnt_lsb));
  119. }
  120. }
  121. static u64 dw_xdata_perf_diff(u64 *m1, u64 *m2, u64 time)
  122. {
  123. u64 rate = (*m1 - *m2);
  124. rate *= (1000 * 1000 * 1000);
  125. rate >>= 20;
  126. rate = DIV_ROUND_CLOSEST_ULL(rate, time);
  127. return rate;
  128. }
  129. static void dw_xdata_perf(struct dw_xdata *dw, u64 *rate, bool write)
  130. {
  131. struct device *dev = &dw->pdev->dev;
  132. u64 data[2], time[2], diff;
  133. mutex_lock(&dw->mutex);
  134. /* First acquisition of current count frames */
  135. writel(0x0, &(__dw_regs(dw)->perf_control));
  136. dw_xdata_perf_meas(dw, &data[0], write);
  137. time[0] = jiffies;
  138. writel((u32)XPERF_CONTROL_ENABLE, &(__dw_regs(dw)->perf_control));
  139. /*
  140. * Wait 100ms between the 1st count frame acquisition and the 2nd
  141. * count frame acquisition, in order to calculate the speed later
  142. */
  143. mdelay(100);
  144. /* Second acquisition of current count frames */
  145. writel(0x0, &(__dw_regs(dw)->perf_control));
  146. dw_xdata_perf_meas(dw, &data[1], write);
  147. time[1] = jiffies;
  148. writel((u32)XPERF_CONTROL_ENABLE, &(__dw_regs(dw)->perf_control));
  149. /*
  150. * Speed calculation
  151. *
  152. * rate = (2nd count frames - 1st count frames) / (time elapsed)
  153. */
  154. diff = jiffies_to_nsecs(time[1] - time[0]);
  155. *rate = dw_xdata_perf_diff(&data[1], &data[0], diff);
  156. mutex_unlock(&dw->mutex);
  157. dev_dbg(dev, "xData: time=%llu us, %s=%llu MB/s\n",
  158. diff, str_write_read(write), *rate);
  159. }
  160. static struct dw_xdata *misc_dev_to_dw(struct miscdevice *misc_dev)
  161. {
  162. return container_of(misc_dev, struct dw_xdata, misc_dev);
  163. }
  164. static ssize_t write_show(struct device *dev, struct device_attribute *attr,
  165. char *buf)
  166. {
  167. struct miscdevice *misc_dev = dev_get_drvdata(dev);
  168. struct dw_xdata *dw = misc_dev_to_dw(misc_dev);
  169. u64 rate;
  170. dw_xdata_perf(dw, &rate, true);
  171. return sysfs_emit(buf, "%llu\n", rate);
  172. }
  173. static ssize_t write_store(struct device *dev, struct device_attribute *attr,
  174. const char *buf, size_t size)
  175. {
  176. struct miscdevice *misc_dev = dev_get_drvdata(dev);
  177. struct dw_xdata *dw = misc_dev_to_dw(misc_dev);
  178. bool enabled;
  179. int ret;
  180. ret = kstrtobool(buf, &enabled);
  181. if (ret < 0)
  182. return ret;
  183. if (enabled) {
  184. dev_dbg(dev, "xData: requested write transfer\n");
  185. dw_xdata_start(dw, true);
  186. } else {
  187. dev_dbg(dev, "xData: requested stop transfer\n");
  188. dw_xdata_stop(dw);
  189. }
  190. return size;
  191. }
  192. static DEVICE_ATTR_RW(write);
  193. static ssize_t read_show(struct device *dev, struct device_attribute *attr,
  194. char *buf)
  195. {
  196. struct miscdevice *misc_dev = dev_get_drvdata(dev);
  197. struct dw_xdata *dw = misc_dev_to_dw(misc_dev);
  198. u64 rate;
  199. dw_xdata_perf(dw, &rate, false);
  200. return sysfs_emit(buf, "%llu\n", rate);
  201. }
  202. static ssize_t read_store(struct device *dev, struct device_attribute *attr,
  203. const char *buf, size_t size)
  204. {
  205. struct miscdevice *misc_dev = dev_get_drvdata(dev);
  206. struct dw_xdata *dw = misc_dev_to_dw(misc_dev);
  207. bool enabled;
  208. int ret;
  209. ret = kstrtobool(buf, &enabled);
  210. if (ret < 0)
  211. return ret;
  212. if (enabled) {
  213. dev_dbg(dev, "xData: requested read transfer\n");
  214. dw_xdata_start(dw, false);
  215. } else {
  216. dev_dbg(dev, "xData: requested stop transfer\n");
  217. dw_xdata_stop(dw);
  218. }
  219. return size;
  220. }
  221. static DEVICE_ATTR_RW(read);
  222. static struct attribute *xdata_attrs[] = {
  223. &dev_attr_write.attr,
  224. &dev_attr_read.attr,
  225. NULL,
  226. };
  227. ATTRIBUTE_GROUPS(xdata);
  228. static int dw_xdata_pcie_probe(struct pci_dev *pdev,
  229. const struct pci_device_id *pid)
  230. {
  231. struct device *dev = &pdev->dev;
  232. struct dw_xdata *dw;
  233. char name[24];
  234. u64 addr;
  235. int err;
  236. int id;
  237. /* Enable PCI device */
  238. err = pcim_enable_device(pdev);
  239. if (err) {
  240. dev_err(dev, "enabling device failed\n");
  241. return err;
  242. }
  243. /* Mapping PCI BAR regions */
  244. err = pcim_iomap_regions(pdev, BIT(BAR_0), pci_name(pdev));
  245. if (err) {
  246. dev_err(dev, "xData BAR I/O remapping failed\n");
  247. return err;
  248. }
  249. pci_set_master(pdev);
  250. /* Allocate memory */
  251. dw = devm_kzalloc(dev, sizeof(*dw), GFP_KERNEL);
  252. if (!dw)
  253. return -ENOMEM;
  254. /* Data structure initialization */
  255. mutex_init(&dw->mutex);
  256. dw->rg_region.vaddr = pcim_iomap_table(pdev)[BAR_0];
  257. if (!dw->rg_region.vaddr)
  258. return -ENOMEM;
  259. dw->rg_region.paddr = pdev->resource[BAR_0].start;
  260. dw->max_wr_len = pcie_get_mps(pdev);
  261. dw->max_wr_len >>= 2;
  262. dw->max_rd_len = pcie_get_readrq(pdev);
  263. dw->max_rd_len >>= 2;
  264. dw->pdev = pdev;
  265. id = ida_alloc(&xdata_ida, GFP_KERNEL);
  266. if (id < 0) {
  267. dev_err(dev, "xData: unable to get id\n");
  268. return id;
  269. }
  270. snprintf(name, sizeof(name), DW_XDATA_DRIVER_NAME ".%d", id);
  271. dw->misc_dev.name = kstrdup(name, GFP_KERNEL);
  272. if (!dw->misc_dev.name) {
  273. err = -ENOMEM;
  274. goto err_ida_remove;
  275. }
  276. dw->misc_dev.minor = MISC_DYNAMIC_MINOR;
  277. dw->misc_dev.parent = dev;
  278. dw->misc_dev.groups = xdata_groups;
  279. writel(0x0, &(__dw_regs(dw)->RAM_addr));
  280. writel(0x0, &(__dw_regs(dw)->RAM_port));
  281. addr = dw->rg_region.paddr + DW_XDATA_EP_MEM_OFFSET;
  282. writel(lower_32_bits(addr), &(__dw_regs(dw)->addr_lsb));
  283. writel(upper_32_bits(addr), &(__dw_regs(dw)->addr_msb));
  284. dev_dbg(dev, "xData: target address = 0x%.16llx\n", addr);
  285. dev_dbg(dev, "xData: wr_len = %zu, rd_len = %zu\n",
  286. dw->max_wr_len * 4, dw->max_rd_len * 4);
  287. /* Saving data structure reference */
  288. pci_set_drvdata(pdev, dw);
  289. /* Register misc device */
  290. err = misc_register(&dw->misc_dev);
  291. if (err) {
  292. dev_err(dev, "xData: failed to register device\n");
  293. goto err_kfree_name;
  294. }
  295. return 0;
  296. err_kfree_name:
  297. kfree(dw->misc_dev.name);
  298. err_ida_remove:
  299. ida_free(&xdata_ida, id);
  300. return err;
  301. }
  302. static void dw_xdata_pcie_remove(struct pci_dev *pdev)
  303. {
  304. struct dw_xdata *dw = pci_get_drvdata(pdev);
  305. int id;
  306. if (sscanf(dw->misc_dev.name, DW_XDATA_DRIVER_NAME ".%d", &id) != 1)
  307. return;
  308. if (id < 0)
  309. return;
  310. dw_xdata_stop(dw);
  311. misc_deregister(&dw->misc_dev);
  312. kfree(dw->misc_dev.name);
  313. ida_free(&xdata_ida, id);
  314. }
  315. static const struct pci_device_id dw_xdata_pcie_id_table[] = {
  316. { PCI_DEVICE_DATA(SYNOPSYS, EDDA, NULL) },
  317. { }
  318. };
  319. MODULE_DEVICE_TABLE(pci, dw_xdata_pcie_id_table);
  320. static struct pci_driver dw_xdata_pcie_driver = {
  321. .name = DW_XDATA_DRIVER_NAME,
  322. .id_table = dw_xdata_pcie_id_table,
  323. .probe = dw_xdata_pcie_probe,
  324. .remove = dw_xdata_pcie_remove,
  325. };
  326. module_pci_driver(dw_xdata_pcie_driver);
  327. MODULE_LICENSE("GPL v2");
  328. MODULE_DESCRIPTION("Synopsys DesignWare xData PCIe driver");
  329. MODULE_AUTHOR("Gustavo Pimentel <gustavo.pimentel@synopsys.com>");