1 /* SPDX-License-Identifier: MIT */
3 * Copyright © 2020 Intel Corporation
6 #ifndef __GEN6_PPGTT_H__
7 #define __GEN6_PPGTT_H__
12 struct i915_ppgtt base
;
16 gen6_pte_t __iomem
*pd_addr
;
19 struct mutex pin_mutex
;
21 bool scan_for_unused_pt
;
24 static inline u32
gen6_pte_index(u32 addr
)
26 return i915_pte_index(addr
, GEN6_PDE_SHIFT
);
29 static inline u32
gen6_pte_count(u32 addr
, u32 length
)
31 return i915_pte_count(addr
, length
, GEN6_PDE_SHIFT
);
34 static inline u32
gen6_pde_index(u32 addr
)
36 return i915_pde_index(addr
, GEN6_PDE_SHIFT
);
39 #define __to_gen6_ppgtt(base) container_of(base, struct gen6_ppgtt, base)
41 static inline struct gen6_ppgtt
*to_gen6_ppgtt(struct i915_ppgtt
*base
)
43 BUILD_BUG_ON(offsetof(struct gen6_ppgtt
, base
));
44 return __to_gen6_ppgtt(base
);
48 * gen6_for_each_pde() iterates over every pde from start until start+length.
49 * If start and start+length are not perfectly divisible, the macro will round
50 * down and up as needed. Start=0 and length=2G effectively iterates over
51 * every PDE in the system. The macro modifies ALL its parameters except 'pd',
52 * so each of the other parameters should preferably be a simple variable, or
53 * at most an lvalue with no side-effects!
55 #define gen6_for_each_pde(pt, pd, start, length, iter) \
56 for (iter = gen6_pde_index(start); \
57 length > 0 && iter < I915_PDES && \
58 (pt = i915_pt_entry(pd, iter), true); \
59 ({ u32 temp = ALIGN(start+1, 1 << GEN6_PDE_SHIFT); \
60 temp = min(temp - start, length); \
61 start += temp, length -= temp; }), ++iter)
63 #define gen6_for_all_pdes(pt, pd, iter) \
66 (pt = i915_pt_entry(pd, iter), true); \
69 int gen6_ppgtt_pin(struct i915_ppgtt
*base
);
70 void gen6_ppgtt_unpin(struct i915_ppgtt
*base
);
71 void gen6_ppgtt_unpin_all(struct i915_ppgtt
*base
);
72 void gen6_ppgtt_enable(struct intel_gt
*gt
);
73 void gen7_ppgtt_enable(struct intel_gt
*gt
);
74 struct i915_ppgtt
*gen6_ppgtt_create(struct intel_gt
*gt
);