4 * DSP-BIOS Bridge driver support functions for TI OMAP processors.
6 * DSP/BIOS Bridge driver interface.
8 * Copyright (C) 2005-2006 Texas Instruments, Inc.
10 * This package is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License version 2 as
12 * published by the Free Software Foundation.
14 * THIS PACKAGE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
15 * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
16 * WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
19 /* ----------------------------------- Host OS */
23 #include <dspbridge/host_os.h>
24 #include <linux/types.h>
25 #include <linux/platform_device.h>
29 #include <linux/module.h>
32 #include <linux/device.h>
33 #include <linux/init.h>
34 #include <linux/moduleparam.h>
35 #include <linux/cdev.h>
37 /* ----------------------------------- DSP/BIOS Bridge */
38 #include <dspbridge/dbdefs.h>
40 /* ----------------------------------- Trace & Debug */
41 #include <dspbridge/dbc.h>
43 /* ----------------------------------- OS Adaptation Layer */
44 #include <dspbridge/clk.h>
45 #include <dspbridge/sync.h>
47 /* ----------------------------------- Platform Manager */
48 #include <dspbridge/dspapi-ioctl.h>
49 #include <dspbridge/dspapi.h>
50 #include <dspbridge/dspdrv.h>
52 /* ----------------------------------- Resource Manager */
53 #include <dspbridge/pwr.h>
55 /* ----------------------------------- This */
56 #include <drv_interface.h>
58 #include <dspbridge/resourcecleanup.h>
59 #include <dspbridge/chnl.h>
60 #include <dspbridge/proc.h>
61 #include <dspbridge/dev.h>
62 #include <dspbridge/drv.h>
64 #ifdef CONFIG_TIDSPBRIDGE_DVFS
65 #include <mach-omap2/omap3-opp.h>
68 /* ----------------------------------- Globals */
69 #define DRIVER_NAME "DspBridge"
70 #define DSPBRIDGE_VERSION "0.3"
73 struct platform_device *omap_dspbridge_dev;
74 struct device *bridge;
76 /* This is a test variable used by Bridge to test different sleep states */
77 s32 dsp_test_sleepstate;
79 static struct cdev bridge_cdev;
81 static struct class *bridge_class;
83 static u32 driver_context;
84 static s32 driver_major;
85 static char *base_img;
87 static s32 shm_size = 0x500000; /* 5 MB */
88 static int tc_wordswapon; /* Default value is always false */
89 #ifdef CONFIG_TIDSPBRIDGE_RECOVERY
90 #define REC_TIMEOUT 5000 /*recovery timeout in msecs */
91 static atomic_t bridge_cref; /* number of bridge open handles */
92 static struct workqueue_struct *bridge_rec_queue;
93 static struct work_struct bridge_recovery_work;
94 static DECLARE_COMPLETION(bridge_comp);
95 static DECLARE_COMPLETION(bridge_open_comp);
100 struct omap34_xx_bridge_suspend_data {
102 wait_queue_head_t suspend_wq;
105 static struct omap34_xx_bridge_suspend_data bridge_suspend_data;
107 static int omap34_xxbridge_suspend_lockout(struct omap34_xx_bridge_suspend_data
110 if ((s)->suspended) {
111 if ((f)->f_flags & O_NONBLOCK)
113 wait_event_interruptible((s)->suspend_wq, (s)->suspended == 0);
119 module_param(dsp_debug, int, 0);
120 MODULE_PARM_DESC(dsp_debug, "Wait after loading DSP image. default = false");
122 module_param(dsp_test_sleepstate, int, 0);
123 MODULE_PARM_DESC(dsp_test_sleepstate, "DSP Sleep state = 0");
125 module_param(base_img, charp, 0);
126 MODULE_PARM_DESC(base_img, "DSP base image, default = NULL");
128 module_param(shm_size, int, 0);
129 MODULE_PARM_DESC(shm_size, "shm size, default = 4 MB, minimum = 64 KB");
131 module_param(tc_wordswapon, int, 0);
132 MODULE_PARM_DESC(tc_wordswapon, "TC Word Swap Option. default = 0");
134 MODULE_AUTHOR("Texas Instruments");
135 MODULE_LICENSE("GPL");
136 MODULE_VERSION(DSPBRIDGE_VERSION);
138 static char *driver_name = DRIVER_NAME;
140 static const struct file_operations bridge_fops = {
142 .release = bridge_release,
143 .unlocked_ioctl = bridge_ioctl,
145 .llseek = noop_llseek,
149 static u32 time_out = 1000;
150 #ifdef CONFIG_TIDSPBRIDGE_DVFS
151 s32 dsp_max_opps = VDD1_OPP5;
154 /* Maximum Opps that can be requested by IVA */
156 #ifdef CONFIG_TIDSPBRIDGE_DVFS
157 const struct omap_opp vdd1_rate_table_bridge[] = {
160 {S125M, VDD1_OPP1, 0},
162 {S250M, VDD1_OPP2, 0},
164 {S500M, VDD1_OPP3, 0},
166 {S550M, VDD1_OPP4, 0},
168 {S600M, VDD1_OPP5, 0},
173 struct omap_dsp_platform_data *omap_dspbridge_pdata;
175 u32 vdd1_dsp_freq[6][4] = {
178 {0, 90000, 0, 86000},
180 {0, 180000, 80000, 170000},
182 {0, 360000, 160000, 340000},
184 {0, 396000, 325000, 376000},
186 {0, 430000, 355000, 430000},
189 #ifdef CONFIG_TIDSPBRIDGE_RECOVERY
190 static void bridge_recover(struct work_struct *work)
192 struct dev_object *dev;
193 struct cfg_devnode *dev_node;
194 if (atomic_read(&bridge_cref)) {
195 INIT_COMPLETION(bridge_comp);
196 while (!wait_for_completion_timeout(&bridge_comp,
197 msecs_to_jiffies(REC_TIMEOUT)))
198 pr_info("%s:%d handle(s) still opened\n",
199 __func__, atomic_read(&bridge_cref));
201 dev = dev_get_first();
202 dev_get_dev_node(dev, &dev_node);
203 if (!dev_node || proc_auto_start(dev_node, dev))
204 pr_err("DSP could not be restarted\n");
206 complete_all(&bridge_open_comp);
209 void bridge_recover_schedule(void)
211 INIT_COMPLETION(bridge_open_comp);
213 queue_work(bridge_rec_queue, &bridge_recovery_work);
216 #ifdef CONFIG_TIDSPBRIDGE_DVFS
217 static int dspbridge_scale_notification(struct notifier_block *op,
218 unsigned long val, void *ptr)
220 struct omap_dsp_platform_data *pdata =
221 omap_dspbridge_dev->dev.platform_data;
223 if (CPUFREQ_POSTCHANGE == val && pdata->dsp_get_opp)
224 pwr_pm_post_scale(PRCM_VDD1, pdata->dsp_get_opp());
229 static struct notifier_block iva_clk_notifier = {
230 .notifier_call = dspbridge_scale_notification,
236 * omap3_bridge_startup() - perform low lever initializations
237 * @pdev: pointer to platform device
239 * Initializes recovery, PM and DVFS required data, before calling
240 * clk and memory init routines.
242 static int omap3_bridge_startup(struct platform_device *pdev)
244 struct omap_dsp_platform_data *pdata = pdev->dev.platform_data;
245 struct drv_data *drv_datap = NULL;
246 u32 phys_membase, phys_memsize;
249 #ifdef CONFIG_TIDSPBRIDGE_RECOVERY
250 bridge_rec_queue = create_workqueue("bridge_rec_queue");
251 INIT_WORK(&bridge_recovery_work, bridge_recover);
252 INIT_COMPLETION(bridge_comp);
256 /* Initialize the wait queue */
257 bridge_suspend_data.suspended = 0;
258 init_waitqueue_head(&bridge_suspend_data.suspend_wq);
260 #ifdef CONFIG_TIDSPBRIDGE_DVFS
261 for (i = 0; i < 6; i++)
262 pdata->mpu_speed[i] = vdd1_rate_table_bridge[i].rate;
264 err = cpufreq_register_notifier(&iva_clk_notifier,
265 CPUFREQ_TRANSITION_NOTIFIER);
267 pr_err("%s: clk_notifier_register failed for iva2_ck\n",
274 drv_datap = kzalloc(sizeof(struct drv_data), GFP_KERNEL);
280 drv_datap->shm_size = shm_size;
281 drv_datap->tc_wordswapon = tc_wordswapon;
284 drv_datap->base_img = kmalloc(strlen(base_img) + 1, GFP_KERNEL);
285 if (!drv_datap->base_img) {
289 strncpy(drv_datap->base_img, base_img, strlen(base_img) + 1);
292 dev_set_drvdata(bridge, drv_datap);
294 if (shm_size < 0x10000) { /* 64 KB */
296 pr_err("%s: shm size must be at least 64 KB\n", __func__);
299 dev_dbg(bridge, "%s: requested shm_size = 0x%x\n", __func__, shm_size);
301 phys_membase = pdata->phys_mempool_base;
302 phys_memsize = pdata->phys_mempool_size;
303 if (phys_membase > 0 && phys_memsize > 0)
304 mem_ext_phys_pool_init(phys_membase, phys_memsize);
307 dev_dbg(bridge, "%s: TC Word Swap is enabled\n", __func__);
309 driver_context = dsp_init(&err);
311 pr_err("DSP Bridge driver initialization failed\n");
318 mem_ext_phys_pool_release();
320 kfree(drv_datap->base_img);
324 #ifdef CONFIG_TIDSPBRIDGE_DVFS
325 cpufreq_unregister_notifier(&iva_clk_notifier,
326 CPUFREQ_TRANSITION_NOTIFIER);
333 static int __devinit omap34_xx_bridge_probe(struct platform_device *pdev)
337 #ifdef CONFIG_TIDSPBRIDGE_DVFS
341 omap_dspbridge_dev = pdev;
343 /* Global bridge device */
344 bridge = &omap_dspbridge_dev->dev;
346 /* Bridge low level initializations */
347 err = omap3_bridge_startup(pdev);
351 /* use 2.6 device model */
352 err = alloc_chrdev_region(&dev, 0, 1, driver_name);
354 pr_err("%s: Can't get major %d\n", __func__, driver_major);
358 cdev_init(&bridge_cdev, &bridge_fops);
359 bridge_cdev.owner = THIS_MODULE;
361 err = cdev_add(&bridge_cdev, dev, 1);
363 pr_err("%s: Failed to add bridge device\n", __func__);
368 bridge_class = class_create(THIS_MODULE, "ti_bridge");
369 if (IS_ERR(bridge_class)) {
370 pr_err("%s: Error creating bridge class\n", __func__);
374 driver_major = MAJOR(dev);
375 device_create(bridge_class, NULL, MKDEV(driver_major, 0),
377 pr_info("DSP Bridge driver loaded\n");
382 cdev_del(&bridge_cdev);
384 unregister_chrdev_region(dev, 1);
389 static int __devexit omap34_xx_bridge_remove(struct platform_device *pdev)
394 struct drv_data *drv_datap = dev_get_drvdata(bridge);
396 /* Retrieve the Object handle from the driver data */
397 if (!drv_datap || !drv_datap->drv_object) {
399 pr_err("%s: Failed to retrieve the object handle\n", __func__);
403 #ifdef CONFIG_TIDSPBRIDGE_DVFS
404 if (cpufreq_unregister_notifier(&iva_clk_notifier,
405 CPUFREQ_TRANSITION_NOTIFIER))
406 pr_err("%s: cpufreq_unregister_notifier failed for iva2_ck\n",
408 #endif /* #ifdef CONFIG_TIDSPBRIDGE_DVFS */
410 if (driver_context) {
411 /* Put the DSP in reset state */
412 ret = dsp_deinit(driver_context);
414 DBC_ASSERT(ret == true);
418 mem_ext_phys_pool_release();
422 devno = MKDEV(driver_major, 0);
423 cdev_del(&bridge_cdev);
424 unregister_chrdev_region(devno, 1);
426 /* remove the device from sysfs */
427 device_destroy(bridge_class, MKDEV(driver_major, 0));
428 class_destroy(bridge_class);
435 static int BRIDGE_SUSPEND(struct platform_device *pdev, pm_message_t state)
438 u32 command = PWR_EMERGENCYDEEPSLEEP;
440 status = pwr_sleep_dsp(command, time_out);
444 bridge_suspend_data.suspended = 1;
448 static int BRIDGE_RESUME(struct platform_device *pdev)
452 status = pwr_wake_dsp(time_out);
456 bridge_suspend_data.suspended = 0;
457 wake_up(&bridge_suspend_data.suspend_wq);
461 #define BRIDGE_SUSPEND NULL
462 #define BRIDGE_RESUME NULL
465 static struct platform_driver bridge_driver = {
469 .probe = omap34_xx_bridge_probe,
470 .remove = __devexit_p(omap34_xx_bridge_remove),
471 .suspend = BRIDGE_SUSPEND,
472 .resume = BRIDGE_RESUME,
475 static int __init bridge_init(void)
477 return platform_driver_register(&bridge_driver);
480 static void __exit bridge_exit(void)
482 platform_driver_unregister(&bridge_driver);
486 * This function is called when an application opens handle to the
489 static int bridge_open(struct inode *ip, struct file *filp)
492 struct process_context *pr_ctxt = NULL;
495 * Allocate a new process context and insert it into global
496 * process context list.
499 #ifdef CONFIG_TIDSPBRIDGE_RECOVERY
501 if (filp->f_flags & O_NONBLOCK ||
502 wait_for_completion_interruptible(&bridge_open_comp))
506 pr_ctxt = kzalloc(sizeof(struct process_context), GFP_KERNEL);
508 pr_ctxt->res_state = PROC_RES_ALLOCATED;
509 spin_lock_init(&pr_ctxt->dmm_map_lock);
510 INIT_LIST_HEAD(&pr_ctxt->dmm_map_list);
511 spin_lock_init(&pr_ctxt->dmm_rsv_lock);
512 INIT_LIST_HEAD(&pr_ctxt->dmm_rsv_list);
514 pr_ctxt->node_id = kzalloc(sizeof(struct idr), GFP_KERNEL);
515 if (pr_ctxt->node_id) {
516 idr_init(pr_ctxt->node_id);
522 pr_ctxt->stream_id = kzalloc(sizeof(struct idr), GFP_KERNEL);
523 if (pr_ctxt->stream_id)
524 idr_init(pr_ctxt->stream_id);
531 filp->private_data = pr_ctxt;
532 #ifdef CONFIG_TIDSPBRIDGE_RECOVERY
534 atomic_inc(&bridge_cref);
540 * This function is called when an application closes handle to the bridge
543 static int bridge_release(struct inode *ip, struct file *filp)
546 struct process_context *pr_ctxt;
548 if (!filp->private_data) {
553 pr_ctxt = filp->private_data;
554 flush_signals(current);
555 drv_remove_all_resources(pr_ctxt);
556 proc_detach(pr_ctxt);
559 filp->private_data = NULL;
562 #ifdef CONFIG_TIDSPBRIDGE_RECOVERY
563 if (!atomic_dec_return(&bridge_cref))
564 complete(&bridge_comp);
569 /* This function provides IO interface to the bridge driver. */
570 static long bridge_ioctl(struct file *filp, unsigned int code,
575 union trapped_args buf_in;
577 DBC_REQUIRE(filp != NULL);
578 #ifdef CONFIG_TIDSPBRIDGE_RECOVERY
585 status = omap34_xxbridge_suspend_lockout(&bridge_suspend_data, filp);
590 if (!filp->private_data) {
595 status = copy_from_user(&buf_in, (union trapped_args *)args,
596 sizeof(union trapped_args));
599 status = api_call_dev_ioctl(code, &buf_in, &retval,
605 dev_dbg(bridge, "%s: IOCTL Failed, code: 0x%x "
606 "status 0x%x\n", __func__, code, status);
616 /* This function maps kernel space memory to user space memory. */
617 static int bridge_mmap(struct file *filp, struct vm_area_struct *vma)
619 u32 offset = vma->vm_pgoff << PAGE_SHIFT;
622 DBC_ASSERT(vma->vm_start < vma->vm_end);
624 vma->vm_flags |= VM_RESERVED | VM_IO;
625 vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);
627 dev_dbg(bridge, "%s: vm filp %p offset %x start %lx end %lx page_prot "
628 "%lx flags %lx\n", __func__, filp, offset,
629 vma->vm_start, vma->vm_end, vma->vm_page_prot, vma->vm_flags);
631 status = remap_pfn_range(vma, vma->vm_start, vma->vm_pgoff,
632 vma->vm_end - vma->vm_start,
640 /* To remove all process resources before removing the process from the
641 * process context list */
642 int drv_remove_all_resources(void *process_ctxt)
645 struct process_context *ctxt = (struct process_context *)process_ctxt;
646 drv_remove_all_strm_res_elements(ctxt);
647 drv_remove_all_node_res_elements(ctxt);
648 drv_remove_all_dmm_res_elements(ctxt);
649 ctxt->res_state = PROC_RES_FREED;
653 /* Bridge driver initialization and de-initialization functions */
654 module_init(bridge_init);
655 module_exit(bridge_exit);