2 * Copyright (C) 2014 Marvell Technology Group Ltd.
4 * Antoine Tenart <antoine.tenart@free-electrons.com>
6 * This file is licensed under the terms of the GNU General Public
7 * License version 2. This program is licensed "as is" without any
8 * warranty of any kind, whether express or implied.
11 #include <linux/clk.h>
12 #include <linux/dma-mapping.h>
13 #include <linux/module.h>
15 #include <linux/of_platform.h>
16 #include <linux/phy/phy.h>
17 #include <linux/platform_device.h>
18 #include <linux/usb/chipidea.h>
19 #include <linux/usb/hcd.h>
20 #include <linux/usb/ulpi.h>
24 struct ci_hdrc_usb2_priv
{
25 struct platform_device
*ci_pdev
;
29 static const struct ci_hdrc_platform_data ci_default_pdata
= {
30 .capoffset
= DEF_CAPOFFSET
,
31 .flags
= CI_HDRC_DISABLE_STREAMING
,
34 static struct ci_hdrc_platform_data ci_zynq_pdata
= {
35 .capoffset
= DEF_CAPOFFSET
,
38 static const struct of_device_id ci_hdrc_usb2_of_match
[] = {
39 { .compatible
= "chipidea,usb2"},
40 { .compatible
= "xlnx,zynq-usb-2.20a", .data
= &ci_zynq_pdata
},
43 MODULE_DEVICE_TABLE(of
, ci_hdrc_usb2_of_match
);
45 static int ci_hdrc_usb2_probe(struct platform_device
*pdev
)
47 struct device
*dev
= &pdev
->dev
;
48 struct ci_hdrc_usb2_priv
*priv
;
49 struct ci_hdrc_platform_data
*ci_pdata
= dev_get_platdata(dev
);
51 const struct of_device_id
*match
;
54 ci_pdata
= devm_kmalloc(dev
, sizeof(*ci_pdata
), GFP_KERNEL
);
55 *ci_pdata
= ci_default_pdata
; /* struct copy */
58 match
= of_match_device(ci_hdrc_usb2_of_match
, &pdev
->dev
);
59 if (match
&& match
->data
) {
61 *ci_pdata
= *(struct ci_hdrc_platform_data
*)match
->data
;
64 priv
= devm_kzalloc(dev
, sizeof(*priv
), GFP_KERNEL
);
68 priv
->clk
= devm_clk_get(dev
, NULL
);
69 if (!IS_ERR(priv
->clk
)) {
70 ret
= clk_prepare_enable(priv
->clk
);
72 dev_err(dev
, "failed to enable the clock: %d\n", ret
);
77 ret
= dma_set_mask_and_coherent(dev
, DMA_BIT_MASK(32));
81 ci_pdata
->name
= dev_name(dev
);
83 priv
->ci_pdev
= ci_hdrc_add_device(dev
, pdev
->resource
,
84 pdev
->num_resources
, ci_pdata
);
85 if (IS_ERR(priv
->ci_pdev
)) {
86 ret
= PTR_ERR(priv
->ci_pdev
);
87 if (ret
!= -EPROBE_DEFER
)
89 "failed to register ci_hdrc platform device: %d\n",
94 platform_set_drvdata(pdev
, priv
);
96 pm_runtime_no_callbacks(dev
);
97 pm_runtime_enable(dev
);
102 if (!IS_ERR(priv
->clk
))
103 clk_disable_unprepare(priv
->clk
);
107 static int ci_hdrc_usb2_remove(struct platform_device
*pdev
)
109 struct ci_hdrc_usb2_priv
*priv
= platform_get_drvdata(pdev
);
111 pm_runtime_disable(&pdev
->dev
);
112 ci_hdrc_remove_device(priv
->ci_pdev
);
113 clk_disable_unprepare(priv
->clk
);
118 static struct platform_driver ci_hdrc_usb2_driver
= {
119 .probe
= ci_hdrc_usb2_probe
,
120 .remove
= ci_hdrc_usb2_remove
,
122 .name
= "chipidea-usb2",
123 .of_match_table
= of_match_ptr(ci_hdrc_usb2_of_match
),
126 module_platform_driver(ci_hdrc_usb2_driver
);
128 MODULE_DESCRIPTION("ChipIdea HDRC USB2 binding for ci13xxx");
129 MODULE_AUTHOR("Antoine Tenart <antoine.tenart@free-electrons.com>");
130 MODULE_LICENSE("GPL");