}
 }
 
+static int setup_ppod_edram(struct adapter *adap)
+{
+       unsigned int param, val;
+       int ret;
+
+       /* Driver sends FW_PARAMS_PARAM_DEV_PPOD_EDRAM read command to check
+        * if firmware supports ppod edram feature or not. If firmware
+        * returns 1, then driver can enable this feature by sending
+        * FW_PARAMS_PARAM_DEV_PPOD_EDRAM write command with value 1 to
+        * enable ppod edram feature.
+        */
+       param = (FW_PARAMS_MNEM_V(FW_PARAMS_MNEM_DEV) |
+               FW_PARAMS_PARAM_X_V(FW_PARAMS_PARAM_DEV_PPOD_EDRAM));
+
+       ret = t4_query_params(adap, adap->mbox, adap->pf, 0, 1, ¶m, &val);
+       if (ret < 0) {
+               dev_warn(adap->pdev_dev,
+                        "querying PPOD_EDRAM support failed: %d\n",
+                        ret);
+               return -1;
+       }
+
+       if (val != 1)
+               return -1;
+
+       ret = t4_set_params(adap, adap->mbox, adap->pf, 0, 1, ¶m, &val);
+       if (ret < 0) {
+               dev_err(adap->pdev_dev,
+                       "setting PPOD_EDRAM failed: %d\n", ret);
+               return -1;
+       }
+       return 0;
+}
+
 /**
  *     cxgb4_write_rss - write the RSS table for a given port
  *     @pi: the port
                dev_err(adapter->pdev_dev,
                        "HMA configuration failed with error %d\n", ret);
 
+       if (is_t6(adapter->params.chip)) {
+               ret = setup_ppod_edram(adapter);
+               if (!ret)
+                       dev_info(adapter->pdev_dev, "Successfully enabled "
+                                "ppod edram feature\n");
+       }
+
        /*
         * And finally tell the firmware to initialize itself using the
         * parameters from the Configuration File.
                        goto bye;
                adap->vres.iscsi.start = val[0];
                adap->vres.iscsi.size = val[1] - val[0] + 1;
+               if (is_t6(adap->params.chip)) {
+                       params[0] = FW_PARAM_PFVF(PPOD_EDRAM_START);
+                       params[1] = FW_PARAM_PFVF(PPOD_EDRAM_END);
+                       ret = t4_query_params(adap, adap->mbox, adap->pf, 0, 2,
+                                             params, val);
+                       if (!ret) {
+                               adap->vres.ppod_edram.start = val[0];
+                               adap->vres.ppod_edram.size =
+                                       val[1] - val[0] + 1;
+
+                               dev_info(adap->pdev_dev,
+                                        "ppod edram start 0x%x end 0x%x size 0x%x\n",
+                                        val[0], val[1],
+                                        adap->vres.ppod_edram.size);
+                       }
+               }
                /* LIO target and cxgb4i initiaitor */
                adap->num_ofld_uld += 2;
        }
 
        struct cxgb4_range ocq;
        struct cxgb4_range key;
        unsigned int ncrypto_fc;
+       struct cxgb4_range ppod_edram;
 };
 
 struct chcr_stats_debug {
 
        FW_PARAMS_PARAM_DEV_TPCHMAP     = 0x1F,
        FW_PARAMS_PARAM_DEV_HMA_SIZE    = 0x20,
        FW_PARAMS_PARAM_DEV_RDMA_WRITE_WITH_IMM = 0x21,
+       FW_PARAMS_PARAM_DEV_PPOD_EDRAM  = 0x23,
        FW_PARAMS_PARAM_DEV_RI_WRITE_CMPL_WR    = 0x24,
        FW_PARAMS_PARAM_DEV_OPAQUE_VIID_SMT_EXTN = 0x27,
        FW_PARAMS_PARAM_DEV_HASHFILTER_WITH_OFLD = 0x28,
        FW_PARAMS_PARAM_PFVF_RAWF_END = 0x37,
        FW_PARAMS_PARAM_PFVF_NCRYPTO_LOOKASIDE = 0x39,
        FW_PARAMS_PARAM_PFVF_PORT_CAPS32 = 0x3A,
+       FW_PARAMS_PARAM_PFVF_PPOD_EDRAM_START = 0x3B,
+       FW_PARAMS_PARAM_PFVF_PPOD_EDRAM_END = 0x3C,
        FW_PARAMS_PARAM_PFVF_LINK_STATE = 0x40,
 };
 
 
        unsigned int cpu;
        int i;
 
+       if (!ppm->pool)
+               return -EINVAL;
+
        cpu = get_cpu();
        pool = per_cpu_ptr(ppm->pool, cpu);
        spin_lock_bh(&pool->lock);
        }
 
        ppm->next = i + count;
-       if (ppm->next >= ppm->bmap_index_max)
+       if (ppm->max_index_in_edram && (ppm->next >= ppm->max_index_in_edram))
+               ppm->next = 0;
+       else if (ppm->next >= ppm->bmap_index_max)
                ppm->next = 0;
 
        spin_unlock_bh(&ppm->map_lock);
 
 int cxgbi_ppm_init(void **ppm_pp, struct net_device *ndev,
                   struct pci_dev *pdev, void *lldev,
-                  struct cxgbi_tag_format *tformat,
-                  unsigned int ppmax,
-                  unsigned int llimit,
-                  unsigned int start,
-                  unsigned int reserve_factor)
+                  struct cxgbi_tag_format *tformat, unsigned int iscsi_size,
+                  unsigned int llimit, unsigned int start,
+                  unsigned int reserve_factor, unsigned int iscsi_edram_start,
+                  unsigned int iscsi_edram_size)
 {
        struct cxgbi_ppm *ppm = (struct cxgbi_ppm *)(*ppm_pp);
        struct cxgbi_ppm_pool *pool = NULL;
-       unsigned int ppmax_pool = 0;
        unsigned int pool_index_max = 0;
-       unsigned int alloc_sz;
+       unsigned int ppmax_pool = 0;
        unsigned int ppod_bmap_size;
+       unsigned int alloc_sz;
+       unsigned int ppmax;
+
+       if (!iscsi_edram_start)
+               iscsi_edram_size = 0;
+
+       if (iscsi_edram_size &&
+           ((iscsi_edram_start + iscsi_edram_size) != start)) {
+               pr_err("iscsi ppod region not contiguous: EDRAM start 0x%x "
+                       "size 0x%x DDR start 0x%x\n",
+                       iscsi_edram_start, iscsi_edram_size, start);
+               return -EINVAL;
+       }
+
+       if (iscsi_edram_size) {
+               reserve_factor = 0;
+               start = iscsi_edram_start;
+       }
+
+       ppmax = (iscsi_edram_size + iscsi_size) >> PPOD_SIZE_SHIFT;
 
        if (ppm) {
                pr_info("ippm: %s, ppm 0x%p,0x%p already initialized, %u/%u.\n",
                        __func__, ppmax, ppmax_pool, ppod_bmap_size, start,
                        end);
        }
+       if (iscsi_edram_size) {
+               unsigned int first_ddr_idx =
+                               iscsi_edram_size >> PPOD_SIZE_SHIFT;
+
+               ppm->max_index_in_edram = first_ddr_idx - 1;
+               bitmap_set(ppm->ppod_bmap, first_ddr_idx, 1);
+               pr_debug("reserved %u ppod in bitmap\n", first_ddr_idx);
+       }
 
        spin_lock_init(&ppm->map_lock);
        kref_init(&ppm->refcnt);
 
        spinlock_t map_lock;            /* ppm map lock */
        unsigned int bmap_index_max;
        unsigned int next;
+       unsigned int max_index_in_edram;
        unsigned long *ppod_bmap;
        struct cxgbi_ppod_data ppod_data[0];
 };
                            unsigned long caller_data);
 int cxgbi_ppm_init(void **ppm_pp, struct net_device *, struct pci_dev *,
                   void *lldev, struct cxgbi_tag_format *,
-                  unsigned int ppmax, unsigned int llimit,
-                  unsigned int start,
-                  unsigned int reserve_factor);
+                  unsigned int iscsi_size, unsigned int llimit,
+                  unsigned int start, unsigned int reserve_factor,
+                  unsigned int edram_start, unsigned int edram_size);
 int cxgbi_ppm_release(struct cxgbi_ppm *ppm);
 void cxgbi_tagmask_check(unsigned int tagmask, struct cxgbi_tag_format *);
 unsigned int cxgbi_tagmask_set(unsigned int ppmax);
 
                tformat.pgsz_order[i] = uinfo.pgsz_factor[i];
        cxgbi_tagmask_check(tagmask, &tformat);
 
-       cxgbi_ddp_ppm_setup(&tdev->ulp_iscsi, cdev, &tformat, ppmax,
-                           uinfo.llimit, uinfo.llimit, 0);
+       err = cxgbi_ddp_ppm_setup(&tdev->ulp_iscsi, cdev, &tformat,
+                                 (uinfo.ulimit - uinfo.llimit + 1),
+                                 uinfo.llimit, uinfo.llimit, 0, 0, 0);
+       if (err)
+               return err;
+
        if (!(cdev->flags & CXGBI_FLAG_DDP_OFF)) {
                uinfo.tagmask = tagmask;
                uinfo.ulimit = uinfo.llimit + (ppmax << PPOD_SIZE_SHIFT);
 
        err = cxgb3i_ddp_init(cdev);
        if (err) {
-               pr_info("0x%p ddp init failed\n", cdev);
+               pr_info("0x%p ddp init failed %d\n", cdev, err);
                goto err_out;
        }
 
 
        struct net_device *ndev = cdev->ports[0];
        struct cxgbi_tag_format tformat;
        unsigned int ppmax;
-       int i;
+       int i, err;
 
        if (!lldi->vr->iscsi.size) {
                pr_warn("%s, iscsi NOT enabled, check config!\n", ndev->name);
                                         & 0xF;
        cxgbi_tagmask_check(lldi->iscsi_tagmask, &tformat);
 
-       cxgbi_ddp_ppm_setup(lldi->iscsi_ppm, cdev, &tformat, ppmax,
-                           lldi->iscsi_llimit, lldi->vr->iscsi.start, 2);
+       pr_info("iscsi_edram.start 0x%x iscsi_edram.size 0x%x",
+               lldi->vr->ppod_edram.start, lldi->vr->ppod_edram.size);
+
+       err = cxgbi_ddp_ppm_setup(lldi->iscsi_ppm, cdev, &tformat,
+                                 lldi->vr->iscsi.size, lldi->iscsi_llimit,
+                                 lldi->vr->iscsi.start, 2,
+                                 lldi->vr->ppod_edram.start,
+                                 lldi->vr->ppod_edram.size);
+
+       if (err < 0)
+               return err;
 
        cdev->csk_ddp_setup_digest = ddp_setup_conn_digest;
        cdev->csk_ddp_setup_pgidx = ddp_setup_conn_pgidx;
 
        rc = cxgb4i_ddp_init(cdev);
        if (rc) {
-               pr_info("t4 0x%p ddp init failed.\n", cdev);
+               pr_info("t4 0x%p ddp init failed %d.\n", cdev, rc);
                goto err_out;
        }
        rc = cxgb4i_ofld_init(cdev);
 
 
 static unsigned char padding[4];
 
-void cxgbi_ddp_ppm_setup(void **ppm_pp, struct cxgbi_device *cdev,
-                        struct cxgbi_tag_format *tformat, unsigned int ppmax,
-                        unsigned int llimit, unsigned int start,
-                        unsigned int rsvd_factor)
+int cxgbi_ddp_ppm_setup(void **ppm_pp, struct cxgbi_device *cdev,
+                       struct cxgbi_tag_format *tformat,
+                       unsigned int iscsi_size, unsigned int llimit,
+                       unsigned int start, unsigned int rsvd_factor,
+                       unsigned int edram_start, unsigned int edram_size)
 {
        int err = cxgbi_ppm_init(ppm_pp, cdev->ports[0], cdev->pdev,
-                               cdev->lldev, tformat, ppmax, llimit, start,
-                               rsvd_factor);
+                               cdev->lldev, tformat, iscsi_size, llimit, start,
+                               rsvd_factor, edram_start, edram_size);
 
        if (err >= 0) {
                struct cxgbi_ppm *ppm = (struct cxgbi_ppm *)(*ppm_pp);
        } else {
                cdev->flags |= CXGBI_FLAG_DDP_OFF;
        }
+
+       return err;
 }
 EXPORT_SYMBOL_GPL(cxgbi_ddp_ppm_setup);
 
 
 void cxgbi_ddp_set_one_ppod(struct cxgbi_pagepod *,
                            struct cxgbi_task_tag_info *,
                            struct scatterlist **sg_pp, unsigned int *sg_off);
-void cxgbi_ddp_ppm_setup(void **ppm_pp, struct cxgbi_device *,
-                        struct cxgbi_tag_format *, unsigned int ppmax,
-                        unsigned int llimit, unsigned int start,
-                        unsigned int rsvd_factor);
+int cxgbi_ddp_ppm_setup(void **ppm_pp, struct cxgbi_device *cdev,
+                       struct cxgbi_tag_format *tformat,
+                       unsigned int iscsi_size, unsigned int llimit,
+                       unsigned int start, unsigned int rsvd_factor,
+                       unsigned int edram_start, unsigned int edram_size);
 #endif /*__LIBCXGBI_H__*/
 
 
        ret = cxgbi_ppm_init(lldi->iscsi_ppm, cdev->lldi.ports[0],
                             cdev->lldi.pdev, &cdev->lldi, &tformat,
-                            ppmax, lldi->iscsi_llimit,
-                            lldi->vr->iscsi.start, 2);
+                            lldi->vr->iscsi.size, lldi->iscsi_llimit,
+                            lldi->vr->iscsi.start, 2,
+                            lldi->vr->ppod_edram.start,
+                            lldi->vr->ppod_edram.size);
        if (ret >= 0) {
                struct cxgbi_ppm *ppm = (struct cxgbi_ppm *)(*lldi->iscsi_ppm);