Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 | // SPDX-License-Identifier: (GPL-2.0-only OR BSD-3-Clause) // // This file is provided under a dual BSD/GPLv2 license. When using or // redistributing this file, you may do so under either license. // // Copyright(c) 2018 Intel Corporation. All rights reserved. // // Author: Keyon Jie <yang.jie@linux.intel.com> // #include <linux/io-64-nonatomic-lo-hi.h> #include <linux/platform_device.h> #include <asm/unaligned.h> #include <sound/soc.h> #include <sound/sof.h> #include "sof-priv.h" #include "ops.h" /* * Register IO * * The sof_io_xyz() wrappers are typically referenced in snd_sof_dsp_ops * structures and cannot be inlined. */ void sof_io_write(struct snd_sof_dev *sdev, void __iomem *addr, u32 value) { writel(value, addr); } EXPORT_SYMBOL(sof_io_write); u32 sof_io_read(struct snd_sof_dev *sdev, void __iomem *addr) { return readl(addr); } EXPORT_SYMBOL(sof_io_read); void sof_io_write64(struct snd_sof_dev *sdev, void __iomem *addr, u64 value) { writeq(value, addr); } EXPORT_SYMBOL(sof_io_write64); u64 sof_io_read64(struct snd_sof_dev *sdev, void __iomem *addr) { return readq(addr); } EXPORT_SYMBOL(sof_io_read64); /* * IPC Mailbox IO */ void sof_mailbox_write(struct snd_sof_dev *sdev, u32 offset, void *message, size_t bytes) { void __iomem *dest = sdev->bar[sdev->mailbox_bar] + offset; memcpy_toio(dest, message, bytes); } EXPORT_SYMBOL(sof_mailbox_write); void sof_mailbox_read(struct snd_sof_dev *sdev, u32 offset, void *message, size_t bytes) { void __iomem *src = sdev->bar[sdev->mailbox_bar] + offset; memcpy_fromio(message, src, bytes); } EXPORT_SYMBOL(sof_mailbox_read); /* * Memory copy. */ int sof_block_write(struct snd_sof_dev *sdev, enum snd_sof_fw_blk_type blk_type, u32 offset, void *src, size_t size) { int bar = snd_sof_dsp_get_bar_index(sdev, blk_type); const u8 *src_byte = src; void __iomem *dest; u32 affected_mask; u32 tmp; int m, n; if (bar < 0) return bar; dest = sdev->bar[bar] + offset; m = size / 4; n = size % 4; /* __iowrite32_copy use 32bit size values so divide by 4 */ __iowrite32_copy(dest, src, m); if (n) { affected_mask = (1 << (8 * n)) - 1; /* first read the 32bit data of dest, then change affected * bytes, and write back to dest. For unaffected bytes, it * should not be changed */ tmp = ioread32(dest + m * 4); tmp &= ~affected_mask; tmp |= *(u32 *)(src_byte + m * 4) & affected_mask; iowrite32(tmp, dest + m * 4); } return 0; } EXPORT_SYMBOL(sof_block_write); int sof_block_read(struct snd_sof_dev *sdev, enum snd_sof_fw_blk_type blk_type, u32 offset, void *dest, size_t size) { int bar = snd_sof_dsp_get_bar_index(sdev, blk_type); if (bar < 0) return bar; memcpy_fromio(dest, sdev->bar[bar] + offset, size); return 0; } EXPORT_SYMBOL(sof_block_read); /* * Generic buffer page table creation. * Take the each physical page address and drop the least significant unused * bits from each (based on PAGE_SIZE). Then pack valid page address bits * into compressed page table. */ int snd_sof_create_page_table(struct device *dev, struct snd_dma_buffer *dmab, unsigned char *page_table, size_t size) { int i, pages; pages = snd_sgbuf_aligned_pages(size); dev_dbg(dev, "generating page table for %p size 0x%zx pages %d\n", dmab->area, size, pages); for (i = 0; i < pages; i++) { /* * The number of valid address bits for each page is 20. * idx determines the byte position within page_table * where the current page's address is stored * in the compressed page_table. * This can be calculated by multiplying the page number by 2.5. */ u32 idx = (5 * i) >> 1; u32 pfn = snd_sgbuf_get_addr(dmab, i * PAGE_SIZE) >> PAGE_SHIFT; u8 *pg_table; dev_vdbg(dev, "pfn i %i idx %d pfn %x\n", i, idx, pfn); pg_table = (u8 *)(page_table + idx); /* * pagetable compression: * byte 0 byte 1 byte 2 byte 3 byte 4 byte 5 * ___________pfn 0__________ __________pfn 1___________ _pfn 2... * .... .... .... .... .... .... .... .... .... .... .... * It is created by: * 1. set current location to 0, PFN index i to 0 * 2. put pfn[i] at current location in Little Endian byte order * 3. calculate an intermediate value as * x = (pfn[i+1] << 4) | (pfn[i] & 0xf) * 4. put x at offset (current location + 2) in LE byte order * 5. increment current location by 5 bytes, increment i by 2 * 6. continue to (2) */ if (i & 1) put_unaligned_le32((pg_table[0] & 0xf) | pfn << 4, pg_table); else put_unaligned_le32(pfn, pg_table); } return pages; } EXPORT_SYMBOL(snd_sof_create_page_table); |