Skip to content

Commit d6473f5

Browse files
ickletursulin
authored andcommitted
drm/i915: Add support for mapping an object page by page
Introduced a new vm specfic callback insert_page() to program a single pte in ggtt or ppgtt. This allows us to map a single page in to the mappable aperture space. This can be iterated over to access the whole object by using space as meagre as page size. v2: Added low level rpm assertions to insert_page routines (Chris) v3: Added POSTING_READ post register write (Tvrtko) v4: Rebase (Ankit) v5: Removed wmb() and FLUSH_CTL from insert_page, caller to take care of it (Chris) v6: insert_page not working correctly without FLSH_CNTL write, added the write again. Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk> Signed-off-by: Ankitprasad Sharma <ankitprasad.r.sharma@intel.com> Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com> Signed-off-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
1 parent 4e50f79 commit d6473f5

File tree

4 files changed

+81
-1
lines changed

4 files changed

+81
-1
lines changed

drivers/char/agp/intel-gtt.c

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -840,6 +840,14 @@ static bool i830_check_flags(unsigned int flags)
840840
return false;
841841
}
842842

843+
void intel_gtt_insert_page(dma_addr_t addr,
844+
unsigned int pg,
845+
unsigned int flags)
846+
{
847+
intel_private.driver->write_entry(addr, pg, flags);
848+
}
849+
EXPORT_SYMBOL(intel_gtt_insert_page);
850+
843851
void intel_gtt_insert_sg_entries(struct sg_table *st,
844852
unsigned int pg_start,
845853
unsigned int flags)

drivers/gpu/drm/i915/i915_gem_gtt.c

Lines changed: 65 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -2355,6 +2355,28 @@ static void gen8_set_pte(void __iomem *addr, gen8_pte_t pte)
23552355
#endif
23562356
}
23572357

2358+
static void gen8_ggtt_insert_page(struct i915_address_space *vm,
2359+
dma_addr_t addr,
2360+
uint64_t offset,
2361+
enum i915_cache_level level,
2362+
u32 unused)
2363+
{
2364+
struct drm_i915_private *dev_priv = to_i915(vm->dev);
2365+
gen8_pte_t __iomem *pte =
2366+
(gen8_pte_t __iomem *)dev_priv->ggtt.gsm +
2367+
(offset >> PAGE_SHIFT);
2368+
int rpm_atomic_seq;
2369+
2370+
rpm_atomic_seq = assert_rpm_atomic_begin(dev_priv);
2371+
2372+
gen8_set_pte(pte, gen8_pte_encode(addr, level, true));
2373+
2374+
I915_WRITE(GFX_FLSH_CNTL_GEN6, GFX_FLSH_CNTL_EN);
2375+
POSTING_READ(GFX_FLSH_CNTL_GEN6);
2376+
2377+
assert_rpm_atomic_end(dev_priv, rpm_atomic_seq);
2378+
}
2379+
23582380
static void gen8_ggtt_insert_entries(struct i915_address_space *vm,
23592381
struct sg_table *st,
23602382
uint64_t start,
@@ -2424,6 +2446,28 @@ static void gen8_ggtt_insert_entries__BKL(struct i915_address_space *vm,
24242446
stop_machine(gen8_ggtt_insert_entries__cb, &arg, NULL);
24252447
}
24262448

2449+
static void gen6_ggtt_insert_page(struct i915_address_space *vm,
2450+
dma_addr_t addr,
2451+
uint64_t offset,
2452+
enum i915_cache_level level,
2453+
u32 flags)
2454+
{
2455+
struct drm_i915_private *dev_priv = to_i915(vm->dev);
2456+
gen6_pte_t __iomem *pte =
2457+
(gen6_pte_t __iomem *)dev_priv->ggtt.gsm +
2458+
(offset >> PAGE_SHIFT);
2459+
int rpm_atomic_seq;
2460+
2461+
rpm_atomic_seq = assert_rpm_atomic_begin(dev_priv);
2462+
2463+
iowrite32(vm->pte_encode(addr, level, true, flags), pte);
2464+
2465+
I915_WRITE(GFX_FLSH_CNTL_GEN6, GFX_FLSH_CNTL_EN);
2466+
POSTING_READ(GFX_FLSH_CNTL_GEN6);
2467+
2468+
assert_rpm_atomic_end(dev_priv, rpm_atomic_seq);
2469+
}
2470+
24272471
/*
24282472
* Binds an object into the global gtt with the specified cache level. The object
24292473
* will be accessible to the GPU via commands whose operands reference offsets
@@ -2543,6 +2587,24 @@ static void gen6_ggtt_clear_range(struct i915_address_space *vm,
25432587
assert_rpm_atomic_end(dev_priv, rpm_atomic_seq);
25442588
}
25452589

2590+
static void i915_ggtt_insert_page(struct i915_address_space *vm,
2591+
dma_addr_t addr,
2592+
uint64_t offset,
2593+
enum i915_cache_level cache_level,
2594+
u32 unused)
2595+
{
2596+
struct drm_i915_private *dev_priv = to_i915(vm->dev);
2597+
unsigned int flags = (cache_level == I915_CACHE_NONE) ?
2598+
AGP_USER_MEMORY : AGP_USER_CACHED_MEMORY;
2599+
int rpm_atomic_seq;
2600+
2601+
rpm_atomic_seq = assert_rpm_atomic_begin(dev_priv);
2602+
2603+
intel_gtt_insert_page(addr, offset >> PAGE_SHIFT, flags);
2604+
2605+
assert_rpm_atomic_end(dev_priv, rpm_atomic_seq);
2606+
}
2607+
25462608
static void i915_ggtt_insert_entries(struct i915_address_space *vm,
25472609
struct sg_table *pages,
25482610
uint64_t start,
@@ -3076,7 +3138,7 @@ static int gen8_gmch_probe(struct i915_ggtt *ggtt)
30763138

30773139
ggtt->base.bind_vma = ggtt_bind_vma;
30783140
ggtt->base.unbind_vma = ggtt_unbind_vma;
3079-
3141+
ggtt->base.insert_page = gen8_ggtt_insert_page;
30803142
ggtt->base.clear_range = nop_clear_range;
30813143
if (!USES_FULL_PPGTT(dev_priv))
30823144
ggtt->base.clear_range = gen8_ggtt_clear_range;
@@ -3116,6 +3178,7 @@ static int gen6_gmch_probe(struct i915_ggtt *ggtt)
31163178
ret = ggtt_probe_common(dev, ggtt->size);
31173179

31183180
ggtt->base.clear_range = gen6_ggtt_clear_range;
3181+
ggtt->base.insert_page = gen6_ggtt_insert_page;
31193182
ggtt->base.insert_entries = gen6_ggtt_insert_entries;
31203183
ggtt->base.bind_vma = ggtt_bind_vma;
31213184
ggtt->base.unbind_vma = ggtt_unbind_vma;
@@ -3147,6 +3210,7 @@ static int i915_gmch_probe(struct i915_ggtt *ggtt)
31473210
&ggtt->mappable_base, &ggtt->mappable_end);
31483211

31493212
ggtt->do_idle_maps = needs_idle_maps(dev_priv->dev);
3213+
ggtt->base.insert_page = i915_ggtt_insert_page;
31503214
ggtt->base.insert_entries = i915_ggtt_insert_entries;
31513215
ggtt->base.clear_range = i915_ggtt_clear_range;
31523216
ggtt->base.bind_vma = ggtt_bind_vma;

drivers/gpu/drm/i915/i915_gem_gtt.h

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -319,6 +319,11 @@ struct i915_address_space {
319319
uint64_t start,
320320
uint64_t length,
321321
bool use_scratch);
322+
void (*insert_page)(struct i915_address_space *vm,
323+
dma_addr_t addr,
324+
uint64_t offset,
325+
enum i915_cache_level cache_level,
326+
u32 flags);
322327
void (*insert_entries)(struct i915_address_space *vm,
323328
struct sg_table *st,
324329
uint64_t start,

include/drm/intel-gtt.h

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,9 @@ void intel_gmch_remove(void);
1313
bool intel_enable_gtt(void);
1414

1515
void intel_gtt_chipset_flush(void);
16+
void intel_gtt_insert_page(dma_addr_t addr,
17+
unsigned int pg,
18+
unsigned int flags);
1619
void intel_gtt_insert_sg_entries(struct sg_table *st,
1720
unsigned int pg_start,
1821
unsigned int flags);

0 commit comments

Comments
 (0)