2 * Intel(R) Trace Hub driver core
4 * Copyright (C) 2014-2015 Intel Corporation.
6 * This program is free software; you can redistribute it and/or modify it
7 * under the terms and conditions of the GNU General Public License,
8 * version 2, as published by the Free Software Foundation.
10 * This program is distributed in the hope it will be useful, but WITHOUT
11 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
16 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
18 #include <linux/types.h>
19 #include <linux/module.h>
20 #include <linux/device.h>
21 #include <linux/sysfs.h>
22 #include <linux/kdev_t.h>
23 #include <linux/debugfs.h>
24 #include <linux/idr.h>
25 #include <linux/pci.h>
26 #include <linux/dma-mapping.h>
31 static DEFINE_IDA(intel_th_ida
);
33 static int intel_th_match(struct device
*dev
, struct device_driver
*driver
)
35 struct intel_th_driver
*thdrv
= to_intel_th_driver(driver
);
36 struct intel_th_device
*thdev
= to_intel_th_device(dev
);
38 if (thdev
->type
== INTEL_TH_SWITCH
&&
39 (!thdrv
->enable
|| !thdrv
->disable
))
42 return !strcmp(thdev
->name
, driver
->name
);
45 static int intel_th_child_remove(struct device
*dev
, void *data
)
47 device_release_driver(dev
);
52 static int intel_th_probe(struct device
*dev
)
54 struct intel_th_driver
*thdrv
= to_intel_th_driver(dev
->driver
);
55 struct intel_th_device
*thdev
= to_intel_th_device(dev
);
56 struct intel_th_driver
*hubdrv
;
57 struct intel_th_device
*hub
= NULL
;
60 if (thdev
->type
== INTEL_TH_SWITCH
)
63 hub
= to_intel_th_device(dev
->parent
);
65 if (!hub
|| !hub
->dev
.driver
)
68 hubdrv
= to_intel_th_driver(hub
->dev
.driver
);
70 ret
= thdrv
->probe(to_intel_th_device(dev
));
74 if (thdev
->type
== INTEL_TH_OUTPUT
&&
75 !intel_th_output_assigned(thdev
))
76 ret
= hubdrv
->assign(hub
, thdev
);
81 static int intel_th_remove(struct device
*dev
)
83 struct intel_th_driver
*thdrv
= to_intel_th_driver(dev
->driver
);
84 struct intel_th_device
*thdev
= to_intel_th_device(dev
);
85 struct intel_th_device
*hub
= to_intel_th_device(dev
->parent
);
88 if (thdev
->type
== INTEL_TH_SWITCH
) {
89 err
= device_for_each_child(dev
, thdev
, intel_th_child_remove
);
96 if (intel_th_output_assigned(thdev
)) {
97 struct intel_th_driver
*hubdrv
=
98 to_intel_th_driver(dev
->parent
->driver
);
101 hubdrv
->unassign(hub
, thdev
);
107 static struct bus_type intel_th_bus
= {
110 .match
= intel_th_match
,
111 .probe
= intel_th_probe
,
112 .remove
= intel_th_remove
,
115 static void intel_th_device_free(struct intel_th_device
*thdev
);
117 static void intel_th_device_release(struct device
*dev
)
119 intel_th_device_free(to_intel_th_device(dev
));
122 static struct device_type intel_th_source_device_type
= {
123 .name
= "intel_th_source_device",
124 .release
= intel_th_device_release
,
127 static char *intel_th_output_devnode(struct device
*dev
, umode_t
*mode
,
128 kuid_t
*uid
, kgid_t
*gid
)
130 struct intel_th_device
*thdev
= to_intel_th_device(dev
);
134 node
= kasprintf(GFP_KERNEL
, "intel_th%d/%s%d", 0, thdev
->name
,
137 node
= kasprintf(GFP_KERNEL
, "intel_th%d/%s", 0, thdev
->name
);
142 static ssize_t
port_show(struct device
*dev
, struct device_attribute
*attr
,
145 struct intel_th_device
*thdev
= to_intel_th_device(dev
);
147 if (thdev
->output
.port
>= 0)
148 return scnprintf(buf
, PAGE_SIZE
, "%u\n", thdev
->output
.port
);
150 return scnprintf(buf
, PAGE_SIZE
, "unassigned\n");
153 static DEVICE_ATTR_RO(port
);
155 static int intel_th_output_activate(struct intel_th_device
*thdev
)
157 struct intel_th_driver
*thdrv
= to_intel_th_driver(thdev
->dev
.driver
);
160 return thdrv
->activate(thdev
);
162 intel_th_trace_enable(thdev
);
167 static void intel_th_output_deactivate(struct intel_th_device
*thdev
)
169 struct intel_th_driver
*thdrv
= to_intel_th_driver(thdev
->dev
.driver
);
171 if (thdrv
->deactivate
)
172 thdrv
->deactivate(thdev
);
174 intel_th_trace_disable(thdev
);
177 static ssize_t
active_show(struct device
*dev
, struct device_attribute
*attr
,
180 struct intel_th_device
*thdev
= to_intel_th_device(dev
);
182 return scnprintf(buf
, PAGE_SIZE
, "%d\n", thdev
->output
.active
);
185 static ssize_t
active_store(struct device
*dev
, struct device_attribute
*attr
,
186 const char *buf
, size_t size
)
188 struct intel_th_device
*thdev
= to_intel_th_device(dev
);
192 ret
= kstrtoul(buf
, 10, &val
);
196 if (!!val
!= thdev
->output
.active
) {
198 ret
= intel_th_output_activate(thdev
);
200 intel_th_output_deactivate(thdev
);
203 return ret
? ret
: size
;
206 static DEVICE_ATTR_RW(active
);
208 static struct attribute
*intel_th_output_attrs
[] = {
210 &dev_attr_active
.attr
,
214 ATTRIBUTE_GROUPS(intel_th_output
);
216 static struct device_type intel_th_output_device_type
= {
217 .name
= "intel_th_output_device",
218 .groups
= intel_th_output_groups
,
219 .release
= intel_th_device_release
,
220 .devnode
= intel_th_output_devnode
,
223 static struct device_type intel_th_switch_device_type
= {
224 .name
= "intel_th_switch_device",
225 .release
= intel_th_device_release
,
228 static struct device_type
*intel_th_device_type
[] = {
229 [INTEL_TH_SOURCE
] = &intel_th_source_device_type
,
230 [INTEL_TH_OUTPUT
] = &intel_th_output_device_type
,
231 [INTEL_TH_SWITCH
] = &intel_th_switch_device_type
,
234 int intel_th_driver_register(struct intel_th_driver
*thdrv
)
236 if (!thdrv
->probe
|| !thdrv
->remove
)
239 thdrv
->driver
.bus
= &intel_th_bus
;
241 return driver_register(&thdrv
->driver
);
243 EXPORT_SYMBOL_GPL(intel_th_driver_register
);
245 void intel_th_driver_unregister(struct intel_th_driver
*thdrv
)
247 driver_unregister(&thdrv
->driver
);
249 EXPORT_SYMBOL_GPL(intel_th_driver_unregister
);
251 static struct intel_th_device
*
252 intel_th_device_alloc(struct intel_th
*th
, unsigned int type
, const char *name
,
255 struct device
*parent
;
256 struct intel_th_device
*thdev
;
258 if (type
== INTEL_TH_SWITCH
)
261 parent
= &th
->hub
->dev
;
263 thdev
= kzalloc(sizeof(*thdev
) + strlen(name
) + 1, GFP_KERNEL
);
270 strcpy(thdev
->name
, name
);
271 device_initialize(&thdev
->dev
);
272 thdev
->dev
.bus
= &intel_th_bus
;
273 thdev
->dev
.type
= intel_th_device_type
[type
];
274 thdev
->dev
.parent
= parent
;
275 thdev
->dev
.dma_mask
= parent
->dma_mask
;
276 thdev
->dev
.dma_parms
= parent
->dma_parms
;
277 dma_set_coherent_mask(&thdev
->dev
, parent
->coherent_dma_mask
);
279 dev_set_name(&thdev
->dev
, "%d-%s%d", th
->id
, name
, id
);
281 dev_set_name(&thdev
->dev
, "%d-%s", th
->id
, name
);
286 static int intel_th_device_add_resources(struct intel_th_device
*thdev
,
287 struct resource
*res
, int nres
)
291 r
= kmemdup(res
, sizeof(*res
) * nres
, GFP_KERNEL
);
296 thdev
->num_resources
= nres
;
301 static void intel_th_device_remove(struct intel_th_device
*thdev
)
303 device_del(&thdev
->dev
);
304 put_device(&thdev
->dev
);
307 static void intel_th_device_free(struct intel_th_device
*thdev
)
309 kfree(thdev
->resource
);
314 * Intel(R) Trace Hub subdevices
316 static struct intel_th_subdevice
{
318 struct resource res
[3];
323 } intel_th_subdevices
[TH_SUBDEVICE_MAX
] = {
328 .start
= REG_GTH_OFFSET
,
329 .end
= REG_GTH_OFFSET
+ REG_GTH_LENGTH
- 1,
330 .flags
= IORESOURCE_MEM
,
334 .type
= INTEL_TH_SWITCH
,
341 .start
= REG_MSU_OFFSET
,
342 .end
= REG_MSU_OFFSET
+ REG_MSU_LENGTH
- 1,
343 .flags
= IORESOURCE_MEM
,
346 .start
= BUF_MSU_OFFSET
,
347 .end
= BUF_MSU_OFFSET
+ BUF_MSU_LENGTH
- 1,
348 .flags
= IORESOURCE_MEM
,
353 .type
= INTEL_TH_OUTPUT
,
360 .start
= REG_MSU_OFFSET
,
361 .end
= REG_MSU_OFFSET
+ REG_MSU_LENGTH
- 1,
362 .flags
= IORESOURCE_MEM
,
365 .start
= BUF_MSU_OFFSET
,
366 .end
= BUF_MSU_OFFSET
+ BUF_MSU_LENGTH
- 1,
367 .flags
= IORESOURCE_MEM
,
372 .type
= INTEL_TH_OUTPUT
,
379 .start
= REG_STH_OFFSET
,
380 .end
= REG_STH_OFFSET
+ REG_STH_LENGTH
- 1,
381 .flags
= IORESOURCE_MEM
,
386 .flags
= IORESOURCE_MEM
,
391 .type
= INTEL_TH_SOURCE
,
397 .start
= REG_PTI_OFFSET
,
398 .end
= REG_PTI_OFFSET
+ REG_PTI_LENGTH
- 1,
399 .flags
= IORESOURCE_MEM
,
404 .type
= INTEL_TH_OUTPUT
,
411 .start
= REG_DCIH_OFFSET
,
412 .end
= REG_DCIH_OFFSET
+ REG_DCIH_LENGTH
- 1,
413 .flags
= IORESOURCE_MEM
,
418 .type
= INTEL_TH_OUTPUT
,
422 static int intel_th_populate(struct intel_th
*th
, struct resource
*devres
,
423 unsigned int ndevres
, int irq
)
425 struct resource res
[3];
426 unsigned int req
= 0;
429 /* create devices for each intel_th_subdevice */
430 for (i
= 0; i
< ARRAY_SIZE(intel_th_subdevices
); i
++) {
431 struct intel_th_subdevice
*subdev
= &intel_th_subdevices
[i
];
432 struct intel_th_device
*thdev
;
435 thdev
= intel_th_device_alloc(th
, subdev
->type
, subdev
->name
,
442 memcpy(res
, subdev
->res
,
443 sizeof(struct resource
) * subdev
->nres
);
445 for (r
= 0; r
< subdev
->nres
; r
++) {
446 int bar
= TH_MMIO_CONFIG
;
449 * Take .end == 0 to mean 'take the whole bar',
450 * .start then tells us which bar it is. Default to
453 if (!res
[r
].end
&& res
[r
].flags
== IORESOURCE_MEM
) {
456 res
[r
].end
= resource_size(&devres
[bar
]) - 1;
459 if (res
[r
].flags
& IORESOURCE_MEM
) {
460 res
[r
].start
+= devres
[bar
].start
;
461 res
[r
].end
+= devres
[bar
].start
;
463 dev_dbg(th
->dev
, "%s:%d @ %pR\n",
464 subdev
->name
, r
, &res
[r
]);
465 } else if (res
[r
].flags
& IORESOURCE_IRQ
) {
470 err
= intel_th_device_add_resources(thdev
, res
, subdev
->nres
);
472 put_device(&thdev
->dev
);
476 if (subdev
->type
== INTEL_TH_OUTPUT
) {
477 thdev
->dev
.devt
= MKDEV(th
->major
, i
);
478 thdev
->output
.type
= subdev
->otype
;
479 thdev
->output
.port
= -1;
482 err
= device_add(&thdev
->dev
);
484 put_device(&thdev
->dev
);
488 /* need switch driver to be loaded to enumerate the rest */
489 if (subdev
->type
== INTEL_TH_SWITCH
&& !req
) {
491 err
= request_module("intel_th_%s", subdev
->name
);
496 th
->thdev
[i
] = thdev
;
502 for (i
-- ; i
>= 0; i
--)
503 intel_th_device_remove(th
->thdev
[i
]);
508 static int match_devt(struct device
*dev
, void *data
)
510 dev_t devt
= (dev_t
)(unsigned long)data
;
512 return dev
->devt
== devt
;
515 static int intel_th_output_open(struct inode
*inode
, struct file
*file
)
517 const struct file_operations
*fops
;
518 struct intel_th_driver
*thdrv
;
522 dev
= bus_find_device(&intel_th_bus
, NULL
,
523 (void *)(unsigned long)inode
->i_rdev
,
525 if (!dev
|| !dev
->driver
)
528 thdrv
= to_intel_th_driver(dev
->driver
);
529 fops
= fops_get(thdrv
->fops
);
533 replace_fops(file
, fops
);
535 file
->private_data
= to_intel_th_device(dev
);
537 if (file
->f_op
->open
) {
538 err
= file
->f_op
->open(inode
, file
);
545 static const struct file_operations intel_th_output_fops
= {
546 .open
= intel_th_output_open
,
547 .llseek
= noop_llseek
,
551 * intel_th_alloc() - allocate a new Intel TH device and its subdevices
552 * @dev: parent device
553 * @devres: parent's resources
554 * @ndevres: number of resources
558 intel_th_alloc(struct device
*dev
, struct resource
*devres
,
559 unsigned int ndevres
, int irq
)
564 th
= kzalloc(sizeof(*th
), GFP_KERNEL
);
566 return ERR_PTR(-ENOMEM
);
568 th
->id
= ida_simple_get(&intel_th_ida
, 0, 0, GFP_KERNEL
);
574 th
->major
= __register_chrdev(0, 0, TH_POSSIBLE_OUTPUTS
,
575 "intel_th/output", &intel_th_output_fops
);
582 err
= intel_th_populate(th
, devres
, ndevres
, irq
);
589 __unregister_chrdev(th
->major
, 0, TH_POSSIBLE_OUTPUTS
,
593 ida_simple_remove(&intel_th_ida
, th
->id
);
600 EXPORT_SYMBOL_GPL(intel_th_alloc
);
602 void intel_th_free(struct intel_th
*th
)
606 for (i
= 0; i
< TH_SUBDEVICE_MAX
; i
++)
607 if (th
->thdev
[i
] != th
->hub
)
608 intel_th_device_remove(th
->thdev
[i
]);
610 intel_th_device_remove(th
->hub
);
612 __unregister_chrdev(th
->major
, 0, TH_POSSIBLE_OUTPUTS
,
615 ida_simple_remove(&intel_th_ida
, th
->id
);
619 EXPORT_SYMBOL_GPL(intel_th_free
);
622 * intel_th_trace_enable() - enable tracing for an output device
623 * @thdev: output device that requests tracing be enabled
625 int intel_th_trace_enable(struct intel_th_device
*thdev
)
627 struct intel_th_device
*hub
= to_intel_th_device(thdev
->dev
.parent
);
628 struct intel_th_driver
*hubdrv
= to_intel_th_driver(hub
->dev
.driver
);
630 if (WARN_ON_ONCE(hub
->type
!= INTEL_TH_SWITCH
))
633 if (WARN_ON_ONCE(thdev
->type
!= INTEL_TH_OUTPUT
))
636 hubdrv
->enable(hub
, &thdev
->output
);
640 EXPORT_SYMBOL_GPL(intel_th_trace_enable
);
643 * intel_th_trace_disable() - disable tracing for an output device
644 * @thdev: output device that requests tracing be disabled
646 int intel_th_trace_disable(struct intel_th_device
*thdev
)
648 struct intel_th_device
*hub
= to_intel_th_device(thdev
->dev
.parent
);
649 struct intel_th_driver
*hubdrv
= to_intel_th_driver(hub
->dev
.driver
);
651 WARN_ON_ONCE(hub
->type
!= INTEL_TH_SWITCH
);
652 if (WARN_ON_ONCE(thdev
->type
!= INTEL_TH_OUTPUT
))
655 hubdrv
->disable(hub
, &thdev
->output
);
659 EXPORT_SYMBOL_GPL(intel_th_trace_disable
);
661 int intel_th_set_output(struct intel_th_device
*thdev
,
664 struct intel_th_device
*hub
= to_intel_th_device(thdev
->dev
.parent
);
665 struct intel_th_driver
*hubdrv
= to_intel_th_driver(hub
->dev
.driver
);
667 if (!hubdrv
->set_output
)
670 return hubdrv
->set_output(hub
, master
);
672 EXPORT_SYMBOL_GPL(intel_th_set_output
);
674 static int __init
intel_th_init(void)
676 intel_th_debug_init();
678 return bus_register(&intel_th_bus
);
680 subsys_initcall(intel_th_init
);
682 static void __exit
intel_th_exit(void)
684 intel_th_debug_done();
686 bus_unregister(&intel_th_bus
);
688 module_exit(intel_th_exit
);
690 MODULE_LICENSE("GPL v2");
691 MODULE_DESCRIPTION("Intel(R) Trace Hub controller driver");
692 MODULE_AUTHOR("Alexander Shishkin <alexander.shishkin@linux.intel.com>");