Re: [PATCH 3/3 v3] iommu/fsl: Freescale PAMU driver and IOMMU API implementation.
From: Tabi Timur-B04825 <hidden>
Date: 2012-10-22 21:18:15
Also in:
linux-iommu, lkml
On Wed, Oct 17, 2012 at 12:32 PM, Varun Sethi [off-list ref] w= rote:
+ * Copyright (C) 2012 Freescale Semiconductor, Inc.
Copyright 2012 Freescale Semiconductor, Inc.
+ * + */ + +#include <linux/init.h> +#include <linux/iommu.h> +#include <linux/slab.h> +#include <linux/module.h> +#include <linux/types.h> +#include <linux/mm.h> +#include <linux/interrupt.h> +#include <linux/device.h> +#include <linux/of_platform.h> +#include <linux/bootmem.h> +#include <linux/genalloc.h> +#include <asm/io.h> +#include <asm/bitops.h> + +#include "fsl_pamu.h" + +/* PAMU bypass enbale register contains control bits for + * enabling bypass mode on each PAMU. + */
/* * Linux multi-line comments * look like this. */
+#define PAMUBYPENR 0x604
Update struct ccsr_guts instead. http://patchwork.ozlabs.org/patch/141649/
+
+/* define indexes for each operation mapping scenario */
+#define OMI_QMAN 0x00
+#define OMI_FMAN 0x01
+#define OMI_QMAN_PRIV 0x02
+#define OMI_CAAM 0x03
+
+static paace_t *ppaact;
+static paace_t *spaact;
+static struct ome *omt;
+unsigned int max_subwindow_count;
+
+struct gen_pool *spaace_pool;
+
+static paace_t *pamu_get_ppaace(int liodn)
+{
+ if (!ppaact) {
+ pr_err("PPAACT doesn't exist\n");
pr_err("fsl-pamu: PPAACT has not been initialized\n");
Make sure ALL pr_xxx() messages in this file start with "fsl-pamu: "
+ return NULL; + } + + return &ppaact[liodn];
Bounds checking?
+}
+
+/** Sets validation bit of PACCE
+ *
+ * @parm[in] liodn PAACT index for desired PAACE
+ *
+ * @return Returns 0 upon success else error code < 0 returned
+ */
+int pamu_enable_liodn(int liodn)
+{
+ paace_t *ppaace;
+
+ ppaace =3D pamu_get_ppaace(liodn);
+ if (!ppaace)
+ return -ENOENT;error message?
+
+ if (!get_bf(ppaace->addr_bitfields, PPAACE_AF_WSE)) {
+ pr_err("liodn %d not configured\n", liodn);
+ return -EINVAL;
+ }
+
+ /* Ensure that all other stores to the ppaace complete first */
+ mb();
+
+ ppaace->addr_bitfields |=3D PAACE_V_VALID;
+ mb();
+
+ return 0;
+}
+
+/** Clears validation bit of PACCE
+ *
+ * @parm[in] liodn PAACT index for desired PAACE
+ *
+ * @return Returns 0 upon success else error code < 0 returnedThis is not proper kerneldoc format.
+ */
+int pamu_disable_liodn(int liodn)
+{
+ paace_t *ppaace;
+
+ ppaace =3D pamu_get_ppaace(liodn);
+ if (!ppaace)
+ return -ENOENT;error message?
+ + set_bf(ppaace->addr_bitfields, PAACE_AF_V, PAACE_V_INVALID); + mb(); + + return 0; +} + + +static unsigned int map_addrspace_size_to_wse(phys_addr_t addrspace_size=
)
+{
+ BUG_ON((addrspace_size & (addrspace_size - 1)));
+
+ /* window size is 2^(WSE+1) bytes */
+ return __ffs(addrspace_size >> PAMU_PAGE_SHIFT) + PAMU_PAGE_SHIFT=- 1;
+}
+
+static unsigned int map_subwindow_cnt_to_wce(u32 subwindow_cnt)
+{
+ /* window count is 2^(WCE+1) bytes */
+ return __ffs(subwindow_cnt) - 1;
+}
+
+static void pamu_setup_default_xfer_to_host_ppaace(paace_t *ppaace)
+{
+ set_bf(ppaace->addr_bitfields, PAACE_AF_PT, PAACE_PT_PRIMARY);
+
+ set_bf(ppaace->domain_attr.to_host.coherency_required, PAACE_DA_H=OST_CR,
+ PAACE_M_COHERENCE_REQ);
+}
+
+static void pamu_setup_default_xfer_to_host_spaace(paace_t *spaace)
+{
+ set_bf(spaace->addr_bitfields, PAACE_AF_PT, PAACE_PT_SECONDARY);
+ set_bf(spaace->domain_attr.to_host.coherency_required, PAACE_DA_H=OST_CR,
+ PAACE_M_COHERENCE_REQ);
+}
+
+static paace_t *pamu_get_spaace(u32 fspi_index, u32 wnum)
+{
+ return &spaact[fspi_index + wnum];bounds checking?
+}
+
+static unsigned long pamu_get_fspi_and_allocate(u32 subwin_cnt)
+{subwin_cnt should probably be an unsigned int. This function needs to be documented. What value is being returned?
+ unsigned long spaace_addr; + + spaace_addr =3D gen_pool_alloc(spaace_pool, subwin_cnt * sizeof(p=
aace_t));
+ if (!spaace_addr) + return ULONG_MAX;
What's wrong with returning 0 on error?
+ + return (spaace_addr - (unsigned long)spaact) / (sizeof(paace_t));
Is this supposed to be a virtual address? If so, then return void* instead of an unsigned long.
+}
+
+void pamu_free_subwins(int liodn)
+{
+ paace_t *ppaace;
+ u32 subwin_cnt, size;subwin_cnt should probably be an unsigned int.
+ + ppaace =3D pamu_get_ppaace(liodn); + if (!ppaace) + return;
error message
+
+ if (get_bf(ppaace->addr_bitfields, PPAACE_AF_MW)) {
+ subwin_cnt =3D 1UL << (get_bf(ppaace->impl_attr, PAACE_IA=_WCE) + 1);
+ size =3D (subwin_cnt - 1) * sizeof(paace_t); + gen_pool_free(spaace_pool, (unsigned long)&spaact[ppaace-= fspi], size); + set_bf(ppaace->addr_bitfields, PPAACE_AF_MW, 0); + } +} + +/* Function used for updating stash destination for the coresspong LIODN=
.
+ */
+int pamu_update_paace_stash(int liodn, u32 subwin, u32 value)
+{
+ paace_t *paace;
+
+ paace =3D pamu_get_ppaace(liodn);
+ if (!paace) {
+ return -ENOENT;
+ }Error message?
+ if (subwin) {
+ paace =3D pamu_get_spaace(paace->fspi, subwin - 1);
+ if (!paace) {
+ return -ENOENT;Error message?
+ } + } + set_bf(paace->impl_attr, PAACE_IA_CID, value); + + return 0; +} + +/** Sets up PPAACE entry for specified liodn + * + * @param[in] liodn Logical IO device number + * @param[in] win_addr starting address of DSA window + * @param[in] win-size size of DSA window + * @param[in] omi Operation mapping index -- if ~omi =3D=3D 0 the=
n omi not defined
+ * @param[in] rpn real (true physical) page number + * @param[in] stashid cache stash id for associated cpu -- if ~stashi=
d =3D=3D 0 then
+ * stashid not defined + * @param[in] snoopid snoop id for hardware coherency -- if ~snoopid =
=3D=3D 0 then
+ * snoopid not defined + * @param[in] subwin_cnt number of sub-windows + * @param[in] prot window permissions + * + * @return Returns 0 upon success else error code < 0 returned + */
Please provide proper kerneldoc comments for all of the functions in this f= ile.
+int pamu_config_ppaace(int liodn, phys_addr_t win_addr, phys_addr_t win_=
size,
+ u32 omi, unsigned long rpn, u32 snoopid, u32 stash=
id,
+ u32 subwin_cnt, int prot)
+{
+ paace_t *ppaace;
+ unsigned long fspi;
+
+ if ((win_size & (win_size - 1)) || win_size < PAMU_PAGE_SIZE) {
+ pr_err("window size too small or not a power of two %llx\=n", win_size);
+ return -EINVAL;
+ }
+
+ if (win_addr & (win_size - 1)) {
+ pr_err("window address is not aligned with window size\n"=);
+ return -EINVAL;
+ }
+
+ ppaace =3D pamu_get_ppaace(liodn);
+ if (!ppaace) {
+ return -ENOENT;
+ }
+
+ /* window size is 2^(WSE+1) bytes */
+ set_bf(ppaace->addr_bitfields, PPAACE_AF_WSE,
+ map_addrspace_size_to_wse(win_size));
+
+ pamu_setup_default_xfer_to_host_ppaace(ppaace);
+
+ ppaace->wbah =3D win_addr >> (PAMU_PAGE_SHIFT + 20);
+ set_bf(ppaace->addr_bitfields, PPAACE_AF_WBAL,
+ (win_addr >> PAMU_PAGE_SHIFT));
+
+ /* set up operation mapping if it's configured */
+ if (omi < OME_NUMBER_ENTRIES) {
+ set_bf(ppaace->impl_attr, PAACE_IA_OTM, PAACE_OTM_INDEXED=);
+ ppaace->op_encode.index_ot.omi =3D omi;
+ } else if (~omi !=3D 0) {
+ pr_err("bad operation mapping index: %d\n", omi);
+ return -EINVAL;
+ }
+
+ /* configure stash id */
+ if (~stashid !=3D 0)
+ set_bf(ppaace->impl_attr, PAACE_IA_CID, stashid);
+
+ /* configure snoop id */
+ if (~snoopid !=3D 0)
+ ppaace->domain_attr.to_host.snpid =3D snoopid;
+
+ if (subwin_cnt) {
+ /* The first entry is in the primary PAACE instead */
+ fspi =3D pamu_get_fspi_and_allocate(subwin_cnt - 1);
+ if (fspi =3D=3D ULONG_MAX) {
+ pr_err("spaace indexes exhausted\n");
+ return -EINVAL;
+ }Indentation problem.
+
+ /* window count is 2^(WCE+1) bytes */
+ set_bf(ppaace->impl_attr, PAACE_IA_WCE,
+ map_subwindow_cnt_to_wce(subwin_cnt));
+ set_bf(ppaace->addr_bitfields, PPAACE_AF_MW, 0x1);
+ ppaace->fspi =3D fspi;
+ } else {
+ set_bf(ppaace->impl_attr, PAACE_IA_ATM, PAACE_ATM_WINDOW_=XLATE);
+ ppaace->twbah =3D rpn >> 20; + set_bf(ppaace->win_bitfields, PAACE_WIN_TWBAL, rpn); + set_bf(ppaace->addr_bitfields, PAACE_AF_AP, prot); + set_bf(ppaace->impl_attr, PAACE_IA_WCE, 0); + set_bf(ppaace->addr_bitfields, PPAACE_AF_MW, 0); + } + mb(); + + return 0; +} + +/** Sets up SPAACE entry for specified subwindow + * + * @param[in] liodn Logical IO device number + * @param[in] subwin_cnt number of sub-windows associated with dma-wind=
ow
+ * @param[in] subwin_addr starting address of subwindow + * @param[in] subwin_size size of subwindow + * @param[in] omi Operation mapping index + * @param[in] rpn real (true physical) page number + * @param[in] snoopid snoop id for hardware coherency -- if ~snoopid=
=3D=3D 0 then
+ * snoopid not defined + * @param[in] stashid cache stash id for associated cpu + * @param[in] enable enable/disable subwindow after reconfiguration + * @param[in] prot sub window permissions + * + * @return Returns 0 upon success else error code < 0 returned + */ +int pamu_config_spaace(int liodn, u32 subwin_cnt, u32 subwin_addr, + phys_addr_t subwin_size, u32 omi, unsigned long rp=
n,
+ u32 snoopid, u32 stashid, int enable, int prot)
+{
+ paace_t *paace;
+ unsigned long fspi;
+
+ /* setup sub-windows */
+ if (!subwin_cnt) {
+ pr_err("Invalid subwindow count\n");
+ return -EINVAL;
+ }
+
+ paace =3D pamu_get_ppaace(liodn);
+ if (subwin_addr > 0 && paace) {
+ fspi =3D paace->fspi;
+ paace =3D pamu_get_spaace(fspi, subwin_addr - 1);
+
+ if (paace && !(paace->addr_bitfields & PAACE_V_VALID)) {
+ pamu_setup_default_xfer_to_host_spaace(paace);
+ set_bf(paace->addr_bitfields, SPAACE_AF_LIODN, li=odn);
+ } + } + + if (!paace) + return -ENOENT;
Error message?
+
+ if (!enable && prot =3D=3D PAACE_AP_PERMS_DENIED) {
+ if (subwin_addr > 0)
+ set_bf(paace->addr_bitfields, PAACE_AF_V,
+ PAACE_V_INVALID);
+ else
+ set_bf(paace->addr_bitfields, PAACE_AF_AP,
+ prot);
+ mb();
+ return 0;
+ }
+
+ if (subwin_size & (subwin_size - 1) || subwin_size < PAMU_PAGE_SI=ZE) {+ pr_err("subwindow size out of range, or not a power of 2\=n");
+ return -EINVAL;
+ }
+
+ if (rpn =3D=3D ULONG_MAX) {
+ pr_err("real page number out of range\n");
+ return -EINVAL;
+ }
+
+ /* window size is 2^(WSE+1) bytes */
+ set_bf(paace->win_bitfields, PAACE_WIN_SWSE,
+ map_addrspace_size_to_wse(subwin_size));
+
+ set_bf(paace->impl_attr, PAACE_IA_ATM, PAACE_ATM_WINDOW_XLATE);
+ paace->twbah =3D rpn >> 20;
+ set_bf(paace->win_bitfields, PAACE_WIN_TWBAL, rpn);
+ set_bf(paace->addr_bitfields, PAACE_AF_AP, prot);
+
+ /* configure snoop id */
+ if (~snoopid !=3D 0)
+ paace->domain_attr.to_host.snpid =3D snoopid;
+
+ /* set up operation mapping if it's configured */
+ if (omi < OME_NUMBER_ENTRIES) {
+ set_bf(paace->impl_attr, PAACE_IA_OTM, PAACE_OTM_INDEXED)=;
+ paace->op_encode.index_ot.omi =3D omi;
+ } else if (~omi !=3D 0) {
+ pr_err("bad operation mapping index: %d\n", omi);
+ return -EINVAL;
+ }
+
+ if (~stashid !=3D 0)
+ set_bf(paace->impl_attr, PAACE_IA_CID, stashid);
+
+ smp_wmb();
+
+ if (enable)
+ paace->addr_bitfields |=3D PAACE_V_VALID;
+
+ mb();
+
+ return 0;
+}
+
+void get_ome_index(u32 *omi_index, struct device *dev)
+{
+ if (of_device_is_compatible(dev->of_node, "fsl,qman-portal"))
+ *omi_index =3D OMI_QMAN;
+ if (of_device_is_compatible(dev->of_node, "fsl,qman"))
+ *omi_index =3D OMI_QMAN_PRIV;
+}Documentation?
+ +u32 get_stash_id(u32 stash_dest_hint, u32 vcpu)
Can we make the stash_id a signed integer, and return -1 as an error?
Making it a u32 is awkward because we keep doing stuff like this:
if (~stashid !=3D 0)
and
return ~(u32)0;
+{
+ const u32 *prop;
+ struct device_node *node;
+ u32 cache_level;
+ int len;
+
+ /* Fastpath, exit early if L3/CPC cache is target for stashing */
+ if (stash_dest_hint =3D=3D IOMMU_ATTR_CACHE_L3) {
+ node =3D of_find_compatible_node(NULL, NULL,
+ "fsl,p4080-l3-cache-controller");
+ if (node) {
if (!node) {
pr_err( "no cpc node" );
return ~(u32)0;
}
+ prop =3D of_get_property(node, "cache-stash-id", =
0);
+ if (!prop) {
+ pr_err("missing cache-stash-id at %s\n", =node->full_name);
+ of_node_put(node);
+ return ~(u32)0;
+ }
+ of_node_put(node);
+ return be32_to_cpup(prop);
+ }
+ return ~(u32)0;
+ }
+
+ for_each_node_by_type(node, "cpu") {
+ prop =3D of_get_property(node, "reg", &len);
+ if (be32_to_cpup(prop) =3D=3D vcpu)
+ break;
+ }
+
+ /* find the hwnode that represents the cache */
+ for (cache_level =3D IOMMU_ATTR_CACHE_L1; cache_level <=3D IOMMU_=ATTR_CACHE_L3; cache_level++) {
Shouldn't this be < IOMMU_ATTR_CACHE_L3, since we already handled the
CPC case above?
+ if (stash_dest_hint =3D=3D cache_level) {
+ prop =3D of_get_property(node, "cache-stash-id", =0);
+ if (!prop) {
+ pr_err("missing cache-stash-id at %s\n", =node->full_name);
+ of_node_put(node);
+ return ~(u32)0;
+ }
+ of_node_put(node);
+ return be32_to_cpup(prop);
+ }
+
+ prop =3D of_get_property(node, "next-level-cache", 0);
+ if (!prop) {
+ pr_err("can't find next-level-cache at %s\n",
+ node->full_name);
+ of_node_put(node);
+ return ~(u32)0; /* can't traverse any further */
+ }
+ of_node_put(node);
+
+ /* advance to next node in cache hierarchy */
+ node =3D of_find_node_by_phandle(*prop);
+ if (!node) {
+ pr_err("bad cpu reference %d\n", vcpu);print the full path of any node that has a broken property
+ return ~(u32)0;
+ }
+ }
+
+ pr_err("stash dest not found for %d on vcpu %d\n",
+ stash_dest_hint, vcpu);
+ return ~(u32)0;
+}
+
+#define QMAN_PAACE 1
+#define QMAN_PORTAL_PAACE 2
+#define BMAN_PAACE 3Documentation?
+
+static void setup_qbman_paace(paace_t *ppaace, int paace_type)
+{
+ switch(paace_type) {
+ case QMAN_PAACE:
+ set_bf(ppaace->impl_attr, PAACE_IA_OTM, PAACE_OTM=_INDEXED);
+ ppaace->op_encode.index_ot.omi =3D OMI_QMAN_PRIV; + /* setup QMAN Private data stashing for the L3 ca=
che */
+ set_bf(ppaace->impl_attr, PAACE_IA_CID, get_stash=
_id(IOMMU_ATTR_CACHE_L3, 0));
+ set_bf(ppaace->domain_attr.to_host.coherency_requ=
ired, PAACE_DA_HOST_CR,
+ 0); + break; + case QMAN_PORTAL_PAACE: + set_bf(ppaace->impl_attr, PAACE_IA_OTM, PAACE_OTM=
_INDEXED);
+ ppaace->op_encode.index_ot.omi =3D OMI_QMAN; + /*Set DQRR and Frame stashing for the L3 cache */ + set_bf(ppaace->impl_attr, PAACE_IA_CID, get_stash=
_id(IOMMU_ATTR_CACHE_L3, 0));
+ break; + case BMAN_PAACE: + set_bf(ppaace->domain_attr.to_host.coherency_requ=
ired, PAACE_DA_HOST_CR,
+ 0); + break; + } +}
Seriously, you need to document these functions.
+
+static void __init setup_omt(struct ome *omt)
+{
+ struct ome *ome;
+
+ /* Configure OMI_QMAN */
+ ome =3D &omt[OMI_QMAN];
+
+ ome->moe[IOE_READ_IDX] =3D EOE_VALID | EOE_READ;
+ ome->moe[IOE_EREAD0_IDX] =3D EOE_VALID | EOE_RSA;
+ ome->moe[IOE_WRITE_IDX] =3D EOE_VALID | EOE_WRITE;
+ ome->moe[IOE_EWRITE0_IDX] =3D EOE_VALID | EOE_WWSAO;
+
+ ome->moe[IOE_DIRECT0_IDX] =3D EOE_VALID | EOE_LDEC;
+ ome->moe[IOE_DIRECT1_IDX] =3D EOE_VALID | EOE_LDECPE;
+
+ /* Configure OMI_FMAN */
+ ome =3D &omt[OMI_FMAN];
+ ome->moe[IOE_READ_IDX] =3D EOE_VALID | EOE_READI;
+ ome->moe[IOE_WRITE_IDX] =3D EOE_VALID | EOE_WRITE;
+
+ /* Configure OMI_QMAN private */
+ ome =3D &omt[OMI_QMAN_PRIV];
+ ome->moe[IOE_READ_IDX] =3D EOE_VALID | EOE_READ;
+ ome->moe[IOE_WRITE_IDX] =3D EOE_VALID | EOE_WRITE;
+ ome->moe[IOE_EREAD0_IDX] =3D EOE_VALID | EOE_RSA;
+ ome->moe[IOE_EWRITE0_IDX] =3D EOE_VALID | EOE_WWSA;
+
+ /* Configure OMI_CAAM */
+ ome =3D &omt[OMI_CAAM];
+ ome->moe[IOE_READ_IDX] =3D EOE_VALID | EOE_READI;
+ ome->moe[IOE_WRITE_IDX] =3D EOE_VALID | EOE_WRITE;
+}
+
+int setup_one_pamu(unsigned long pamu_reg_base, unsigned long pamu_reg_s=ize,
+ phys_addr_t ppaact_phys, phys_addr_t spaact_phys,
+ phys_addr_t omt_phys)
+{
+ u32 *pc;
+ struct pamu_mmap_regs *pamu_regs;
+ u32 pc3_val;
+
+ pc3_val =3D in_be32((u32 *)(pamu_reg_base + PAMU_PC3));pamu_reg_base should be a void*. Or even better, create a struct.
+ +#define PAMU_PAGE_SHIFT 12 +#define PAMU_PAGE_SIZE 4096ULL
4096ULL? Why not just 4096?
+ +/* This bitmap advertises the page sizes supported by PAMU hardware + * to the IOMMU API. + */ +#define FSL_PAMU_PGSIZES (~0xFFFUL)
There should be a better way to define this. ~(PAMU_PAGE_SIZE-1) maybe?
+
+static int map_liodn(int liodn, struct fsl_dma_domain *dma_domain)
+{
+ u32 subwin_cnt =3D dma_domain->subwin_cnt;
+ unsigned long rpn;
+ int ret =3D 0, i;
+
+ if (subwin_cnt) {
+ struct dma_subwindow *sub_win_ptr =3D
+ &dma_domain->sub_win_arr[0];
+ for (i =3D 0; i < subwin_cnt; i++) {
+ if (sub_win_ptr[i].valid) {
+ rpn =3D sub_win_ptr[i].paddr >>
+ PAMU_PAGE_SHIFT,
+ spin_lock(&iommu_lock);
+ ret =3D pamu_config_spaace(liodn, subwin_=cnt, i,
+ sub_win_ptr[i].s=
ize,
+ -1, + rpn, + dma_domain->snoo=
p_id,
+ dma_domain->stas=
h_id,
+ (i > 0) ? 1 : 0, + sub_win_ptr[i].p=
rot);
+ spin_unlock(&iommu_lock);
+ if (ret) {
+ pr_err("PAMU SPAACE configuration=failed for liodn %d\n",
+ liodn);
+ return ret;
+ }
+ }
+ }
+ } else {
+Blank line
+
+
+static struct fsl_dma_domain *iommu_alloc_dma_domain(void)
+{
+ struct fsl_dma_domain *domain;
+
+ domain =3D kmem_cache_zalloc(fsl_pamu_domain_cache, GFP_KERNEL);
+ if (!domain)
+ return NULL;
+
+ domain->stash_id =3D -1;
+ domain->snoop_id =3D -1;Here, stash_id is set to -1, but in other places, you use ~0. Please be consistent.
+
+ INIT_LIST_HEAD(&domain->devices);
+
+ spin_lock_init(&domain->domain_lock);
+
+ return domain;
+}
+
+static inline struct fsl_dma_domain *find_domain(struct device *dev)
+{
+ return dev->archdata.iommu_domain;
+}
+
+static void remove_domain_ref(struct device_domain_info *info, u32 subwi=n_cnt)
+{
+ list_del(&info->link);
+ spin_lock(&iommu_lock);
+ if (subwin_cnt)
+ pamu_free_subwins(info->liodn);
+ pamu_disable_liodn(info->liodn);
+ spin_unlock(&iommu_lock);
+ spin_lock(&device_domain_lock);
+ info->dev->archdata.iommu_domain =3D NULL;
+ free_devinfo_mem(info);
+ spin_unlock(&device_domain_lock);
+}
+
+static void destroy_domain(struct fsl_dma_domain *dma_domain)
+{
+ struct device_domain_info *info;
++ while (!list_empty(&dma_domain->devices)) {
+ info =3D list_entry(dma_domain->devices.next,
+ struct device_domain_info, link);
+ remove_domain_ref(info, dma_domain->subwin_cnt);
+ }I wonder if you should use list_for_each_safe() instead.
+ +static int get_subwin_cnt(dma_addr_t geom_size, u32 subwin, u32 *subwin_=
cnt)
+{
+blank line
+ switch (subwin) {
+ case 0:
+ /* We can't support geometry size > 1MB*/
+ if (geom_size !=3D 1024 * 1024)Instead of doing 1024*1024, use math that reflects the hardware limitation of the PAMU: 256 * PAMU_PAGE_SIZE. Create a macro for 256, like PAMU_MAX_SUBWINDOWS or something.
+ return 0; + *subwin_cnt =3D 256; + break; + case 1: + /* No subwindows only a single PAMU window */ + *subwin_cnt =3D 0; + break; + default: + if (subwin > max_subwindow_count || + (subwin & (subwin - 1))) + return 0; + *subwin_cnt =3D subwin; + } + return 1; +} + +static int configure_domain_geometry(struct iommu_domain *domain, void =
*data)
+{
+ int ret =3D 0;I don't think you need to initialize 'ret'
+} + +static int configure_domain_dma_state(struct fsl_dma_domain *dma_domain,=
int enable) bool enable Finally, please CC: me on all IOMMU and PAMU patches you post upstream. --=20 Timur Tabi Linux kernel developer at Freescale=