Skip to content

Commit f776ae6

Browse files
Sean Andersongregkh
authored andcommitted
dma-mapping: use trace_dma_alloc for dma_alloc* instead of using trace_dma_map
[ Upstream commit c4484ab ] In some cases, we use trace_dma_map to trace dma_alloc* functions. This generally follows dma_debug. However, this does not record all of the relevant information for allocations, such as GFP flags. Create new dma_alloc tracepoints for these functions. Note that while dma_alloc_noncontiguous may allocate discontiguous pages (from the CPU's point of view), the device will only see one contiguous mapping. Therefore, we just need to trace dma_addr and size. Signed-off-by: Sean Anderson <[email protected]> Reviewed-by: Steven Rostedt (Google) <[email protected]> Signed-off-by: Christoph Hellwig <[email protected]> Stable-dep-of: 7e2368a ("dma-debug: don't enforce dma mapping check on noncoherent allocations") Signed-off-by: Sasha Levin <[email protected]>
1 parent 1edd532 commit f776ae6

File tree

2 files changed

+102
-7
lines changed

2 files changed

+102
-7
lines changed

include/trace/events/dma.h

Lines changed: 97 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -114,7 +114,7 @@ DEFINE_EVENT(dma_unmap, dma_unmap_resource,
114114
enum dma_data_direction dir, unsigned long attrs),
115115
TP_ARGS(dev, addr, size, dir, attrs));
116116

117-
TRACE_EVENT(dma_alloc,
117+
DECLARE_EVENT_CLASS(dma_alloc_class,
118118
TP_PROTO(struct device *dev, void *virt_addr, dma_addr_t dma_addr,
119119
size_t size, enum dma_data_direction dir, gfp_t flags,
120120
unsigned long attrs),
@@ -149,7 +149,58 @@ TRACE_EVENT(dma_alloc,
149149
decode_dma_attrs(__entry->attrs))
150150
);
151151

152-
TRACE_EVENT(dma_free,
152+
#define DEFINE_ALLOC_EVENT(name) \
153+
DEFINE_EVENT(dma_alloc_class, name, \
154+
TP_PROTO(struct device *dev, void *virt_addr, dma_addr_t dma_addr, \
155+
size_t size, enum dma_data_direction dir, gfp_t flags, \
156+
unsigned long attrs), \
157+
TP_ARGS(dev, virt_addr, dma_addr, size, dir, flags, attrs))
158+
159+
DEFINE_ALLOC_EVENT(dma_alloc);
160+
DEFINE_ALLOC_EVENT(dma_alloc_pages);
161+
162+
TRACE_EVENT(dma_alloc_sgt,
163+
TP_PROTO(struct device *dev, struct sg_table *sgt, size_t size,
164+
enum dma_data_direction dir, gfp_t flags, unsigned long attrs),
165+
TP_ARGS(dev, sgt, size, dir, flags, attrs),
166+
167+
TP_STRUCT__entry(
168+
__string(device, dev_name(dev))
169+
__dynamic_array(u64, phys_addrs, sgt->orig_nents)
170+
__field(u64, dma_addr)
171+
__field(size_t, size)
172+
__field(enum dma_data_direction, dir)
173+
__field(gfp_t, flags)
174+
__field(unsigned long, attrs)
175+
),
176+
177+
TP_fast_assign(
178+
struct scatterlist *sg;
179+
int i;
180+
181+
__assign_str(device);
182+
for_each_sg(sgt->sgl, sg, sgt->orig_nents, i)
183+
((u64 *)__get_dynamic_array(phys_addrs))[i] = sg_phys(sg);
184+
__entry->dma_addr = sg_dma_address(sgt->sgl);
185+
__entry->size = size;
186+
__entry->dir = dir;
187+
__entry->flags = flags;
188+
__entry->attrs = attrs;
189+
),
190+
191+
TP_printk("%s dir=%s dma_addr=%llx size=%zu phys_addrs=%s flags=%s attrs=%s",
192+
__get_str(device),
193+
decode_dma_data_direction(__entry->dir),
194+
__entry->dma_addr,
195+
__entry->size,
196+
__print_array(__get_dynamic_array(phys_addrs),
197+
__get_dynamic_array_len(phys_addrs) /
198+
sizeof(u64), sizeof(u64)),
199+
show_gfp_flags(__entry->flags),
200+
decode_dma_attrs(__entry->attrs))
201+
);
202+
203+
DECLARE_EVENT_CLASS(dma_free_class,
153204
TP_PROTO(struct device *dev, void *virt_addr, dma_addr_t dma_addr,
154205
size_t size, enum dma_data_direction dir, unsigned long attrs),
155206
TP_ARGS(dev, virt_addr, dma_addr, size, dir, attrs),
@@ -181,6 +232,50 @@ TRACE_EVENT(dma_free,
181232
decode_dma_attrs(__entry->attrs))
182233
);
183234

235+
#define DEFINE_FREE_EVENT(name) \
236+
DEFINE_EVENT(dma_free_class, name, \
237+
TP_PROTO(struct device *dev, void *virt_addr, dma_addr_t dma_addr, \
238+
size_t size, enum dma_data_direction dir, unsigned long attrs), \
239+
TP_ARGS(dev, virt_addr, dma_addr, size, dir, attrs))
240+
241+
DEFINE_FREE_EVENT(dma_free);
242+
DEFINE_FREE_EVENT(dma_free_pages);
243+
244+
TRACE_EVENT(dma_free_sgt,
245+
TP_PROTO(struct device *dev, struct sg_table *sgt, size_t size,
246+
enum dma_data_direction dir),
247+
TP_ARGS(dev, sgt, size, dir),
248+
249+
TP_STRUCT__entry(
250+
__string(device, dev_name(dev))
251+
__dynamic_array(u64, phys_addrs, sgt->orig_nents)
252+
__field(u64, dma_addr)
253+
__field(size_t, size)
254+
__field(enum dma_data_direction, dir)
255+
),
256+
257+
TP_fast_assign(
258+
struct scatterlist *sg;
259+
int i;
260+
261+
__assign_str(device);
262+
for_each_sg(sgt->sgl, sg, sgt->orig_nents, i)
263+
((u64 *)__get_dynamic_array(phys_addrs))[i] = sg_phys(sg);
264+
__entry->dma_addr = sg_dma_address(sgt->sgl);
265+
__entry->size = size;
266+
__entry->dir = dir;
267+
),
268+
269+
TP_printk("%s dir=%s dma_addr=%llx size=%zu phys_addrs=%s",
270+
__get_str(device),
271+
decode_dma_data_direction(__entry->dir),
272+
__entry->dma_addr,
273+
__entry->size,
274+
__print_array(__get_dynamic_array(phys_addrs),
275+
__get_dynamic_array_len(phys_addrs) /
276+
sizeof(u64), sizeof(u64)))
277+
);
278+
184279
TRACE_EVENT(dma_map_sg,
185280
TP_PROTO(struct device *dev, struct scatterlist *sgl, int nents,
186281
int ents, enum dma_data_direction dir, unsigned long attrs),

kernel/dma/mapping.c

Lines changed: 5 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -685,8 +685,8 @@ struct page *dma_alloc_pages(struct device *dev, size_t size,
685685
struct page *page = __dma_alloc_pages(dev, size, dma_handle, dir, gfp);
686686

687687
if (page) {
688-
trace_dma_map_page(dev, page_to_phys(page), *dma_handle, size,
689-
dir, 0);
688+
trace_dma_alloc_pages(dev, page_to_virt(page), *dma_handle,
689+
size, dir, gfp, 0);
690690
debug_dma_map_page(dev, page, 0, size, dir, *dma_handle, 0);
691691
}
692692
return page;
@@ -710,7 +710,7 @@ static void __dma_free_pages(struct device *dev, size_t size, struct page *page,
710710
void dma_free_pages(struct device *dev, size_t size, struct page *page,
711711
dma_addr_t dma_handle, enum dma_data_direction dir)
712712
{
713-
trace_dma_unmap_page(dev, dma_handle, size, dir, 0);
713+
trace_dma_free_pages(dev, page_to_virt(page), dma_handle, size, dir, 0);
714714
debug_dma_unmap_page(dev, dma_handle, size, dir);
715715
__dma_free_pages(dev, size, page, dma_handle, dir);
716716
}
@@ -770,7 +770,7 @@ struct sg_table *dma_alloc_noncontiguous(struct device *dev, size_t size,
770770

771771
if (sgt) {
772772
sgt->nents = 1;
773-
trace_dma_map_sg(dev, sgt->sgl, sgt->orig_nents, 1, dir, attrs);
773+
trace_dma_alloc_sgt(dev, sgt, size, dir, gfp, attrs);
774774
debug_dma_map_sg(dev, sgt->sgl, sgt->orig_nents, 1, dir, attrs);
775775
}
776776
return sgt;
@@ -789,7 +789,7 @@ static void free_single_sgt(struct device *dev, size_t size,
789789
void dma_free_noncontiguous(struct device *dev, size_t size,
790790
struct sg_table *sgt, enum dma_data_direction dir)
791791
{
792-
trace_dma_unmap_sg(dev, sgt->sgl, sgt->orig_nents, dir, 0);
792+
trace_dma_free_sgt(dev, sgt, size, dir);
793793
debug_dma_unmap_sg(dev, sgt->sgl, sgt->orig_nents, dir);
794794

795795
if (use_dma_iommu(dev))

0 commit comments

Comments
 (0)