]>
Commit | Line | Data |
---|---|---|
c073f666 | 1 | #include <linux/module.h> |
2 | #include <linux/netdevice.h> | |
3 | #include <linux/mii.h> | |
4 | #include <linux/usb.h> | |
5 | #include <linux/usb/cdc.h> | |
6 | #include <linux/usb/usbnet.h> | |
7 | ||
8 | #define RTL815x_REQT_READ 0xc0 | |
9 | #define RTL815x_REQT_WRITE 0x40 | |
10 | #define RTL815x_REQ_GET_REGS 0x05 | |
11 | #define RTL815x_REQ_SET_REGS 0x05 | |
12 | ||
13 | #define MCU_TYPE_PLA 0x0100 | |
14 | #define OCP_BASE 0xe86c | |
15 | #define BASE_MII 0xa400 | |
16 | ||
17 | #define BYTE_EN_DWORD 0xff | |
18 | #define BYTE_EN_WORD 0x33 | |
19 | #define BYTE_EN_BYTE 0x11 | |
20 | ||
21 | #define R815x_PHY_ID 32 | |
22 | #define REALTEK_VENDOR_ID 0x0bda | |
23 | ||
24 | ||
25 | static int pla_read_word(struct usb_device *udev, u16 index) | |
26 | { | |
b2f47377 | 27 | int ret; |
c073f666 | 28 | u8 shift = index & 2; |
b2f47377 | 29 | __le32 *tmp; |
30 | ||
31 | tmp = kmalloc(sizeof(*tmp), GFP_KERNEL); | |
32 | if (!tmp) | |
33 | return -ENOMEM; | |
c073f666 | 34 | |
35 | index &= ~3; | |
36 | ||
37 | ret = usb_control_msg(udev, usb_rcvctrlpipe(udev, 0), | |
38 | RTL815x_REQ_GET_REGS, RTL815x_REQT_READ, | |
b2f47377 | 39 | index, MCU_TYPE_PLA, tmp, sizeof(*tmp), 500); |
c073f666 | 40 | if (ret < 0) |
b2f47377 | 41 | goto out2; |
c073f666 | 42 | |
b2f47377 | 43 | ret = __le32_to_cpu(*tmp); |
44 | ret >>= (shift * 8); | |
45 | ret &= 0xffff; | |
c073f666 | 46 | |
b2f47377 | 47 | out2: |
48 | kfree(tmp); | |
49 | return ret; | |
c073f666 | 50 | } |
51 | ||
52 | static int pla_write_word(struct usb_device *udev, u16 index, u32 data) | |
53 | { | |
b2f47377 | 54 | __le32 *tmp; |
e7638524 | 55 | u32 mask = 0xffff; |
c073f666 | 56 | u16 byen = BYTE_EN_WORD; |
57 | u8 shift = index & 2; | |
58 | int ret; | |
59 | ||
b2f47377 | 60 | tmp = kmalloc(sizeof(*tmp), GFP_KERNEL); |
61 | if (!tmp) | |
62 | return -ENOMEM; | |
63 | ||
c073f666 | 64 | data &= mask; |
65 | ||
66 | if (shift) { | |
67 | byen <<= shift; | |
68 | mask <<= (shift * 8); | |
69 | data <<= (shift * 8); | |
70 | index &= ~3; | |
71 | } | |
72 | ||
73 | ret = usb_control_msg(udev, usb_rcvctrlpipe(udev, 0), | |
74 | RTL815x_REQ_GET_REGS, RTL815x_REQT_READ, | |
b2f47377 | 75 | index, MCU_TYPE_PLA, tmp, sizeof(*tmp), 500); |
c073f666 | 76 | if (ret < 0) |
b2f47377 | 77 | goto out3; |
c073f666 | 78 | |
b2f47377 | 79 | data |= __le32_to_cpu(*tmp) & ~mask; |
80 | *tmp = __cpu_to_le32(data); | |
c073f666 | 81 | |
82 | ret = usb_control_msg(udev, usb_sndctrlpipe(udev, 0), | |
83 | RTL815x_REQ_SET_REGS, RTL815x_REQT_WRITE, | |
b2f47377 | 84 | index, MCU_TYPE_PLA | byen, tmp, sizeof(*tmp), |
85 | 500); | |
c073f666 | 86 | |
b2f47377 | 87 | out3: |
88 | kfree(tmp); | |
c073f666 | 89 | return ret; |
90 | } | |
91 | ||
92 | static int ocp_reg_read(struct usbnet *dev, u16 addr) | |
93 | { | |
94 | u16 ocp_base, ocp_index; | |
95 | int ret; | |
96 | ||
97 | ocp_base = addr & 0xf000; | |
98 | ret = pla_write_word(dev->udev, OCP_BASE, ocp_base); | |
99 | if (ret < 0) | |
100 | goto out; | |
101 | ||
102 | ocp_index = (addr & 0x0fff) | 0xb000; | |
103 | ret = pla_read_word(dev->udev, ocp_index); | |
104 | ||
105 | out: | |
106 | return ret; | |
107 | } | |
108 | ||
109 | static int ocp_reg_write(struct usbnet *dev, u16 addr, u16 data) | |
110 | { | |
111 | u16 ocp_base, ocp_index; | |
112 | int ret; | |
113 | ||
114 | ocp_base = addr & 0xf000; | |
115 | ret = pla_write_word(dev->udev, OCP_BASE, ocp_base); | |
116 | if (ret < 0) | |
117 | goto out1; | |
118 | ||
119 | ocp_index = (addr & 0x0fff) | 0xb000; | |
120 | ret = pla_write_word(dev->udev, ocp_index, data); | |
121 | ||
122 | out1: | |
123 | return ret; | |
124 | } | |
125 | ||
126 | static int r815x_mdio_read(struct net_device *netdev, int phy_id, int reg) | |
127 | { | |
128 | struct usbnet *dev = netdev_priv(netdev); | |
b771721a | 129 | int ret; |
c073f666 | 130 | |
131 | if (phy_id != R815x_PHY_ID) | |
132 | return -EINVAL; | |
133 | ||
b771721a | 134 | if (usb_autopm_get_interface(dev->intf) < 0) |
135 | return -ENODEV; | |
136 | ||
137 | ret = ocp_reg_read(dev, BASE_MII + reg * 2); | |
138 | ||
139 | usb_autopm_put_interface(dev->intf); | |
140 | return ret; | |
c073f666 | 141 | } |
142 | ||
143 | static | |
144 | void r815x_mdio_write(struct net_device *netdev, int phy_id, int reg, int val) | |
145 | { | |
146 | struct usbnet *dev = netdev_priv(netdev); | |
147 | ||
148 | if (phy_id != R815x_PHY_ID) | |
149 | return; | |
150 | ||
b771721a | 151 | if (usb_autopm_get_interface(dev->intf) < 0) |
152 | return; | |
153 | ||
c073f666 | 154 | ocp_reg_write(dev, BASE_MII + reg * 2, val); |
b771721a | 155 | |
156 | usb_autopm_put_interface(dev->intf); | |
c073f666 | 157 | } |
158 | ||
159 | static int r8153_bind(struct usbnet *dev, struct usb_interface *intf) | |
160 | { | |
161 | int status; | |
162 | ||
163 | status = usbnet_cdc_bind(dev, intf); | |
164 | if (status < 0) | |
165 | return status; | |
166 | ||
167 | dev->mii.dev = dev->net; | |
168 | dev->mii.mdio_read = r815x_mdio_read; | |
169 | dev->mii.mdio_write = r815x_mdio_write; | |
170 | dev->mii.phy_id_mask = 0x3f; | |
171 | dev->mii.reg_num_mask = 0x1f; | |
172 | dev->mii.phy_id = R815x_PHY_ID; | |
173 | dev->mii.supports_gmii = 1; | |
174 | ||
543ae7f9 | 175 | return status; |
c073f666 | 176 | } |
177 | ||
178 | static int r8152_bind(struct usbnet *dev, struct usb_interface *intf) | |
179 | { | |
180 | int status; | |
181 | ||
182 | status = usbnet_cdc_bind(dev, intf); | |
183 | if (status < 0) | |
184 | return status; | |
185 | ||
186 | dev->mii.dev = dev->net; | |
187 | dev->mii.mdio_read = r815x_mdio_read; | |
188 | dev->mii.mdio_write = r815x_mdio_write; | |
189 | dev->mii.phy_id_mask = 0x3f; | |
190 | dev->mii.reg_num_mask = 0x1f; | |
191 | dev->mii.phy_id = R815x_PHY_ID; | |
192 | dev->mii.supports_gmii = 0; | |
193 | ||
543ae7f9 | 194 | return status; |
c073f666 | 195 | } |
196 | ||
197 | static const struct driver_info r8152_info = { | |
198 | .description = "RTL8152 ECM Device", | |
199 | .flags = FLAG_ETHER | FLAG_POINTTOPOINT, | |
200 | .bind = r8152_bind, | |
201 | .unbind = usbnet_cdc_unbind, | |
202 | .status = usbnet_cdc_status, | |
203 | .manage_power = usbnet_manage_power, | |
204 | }; | |
205 | ||
206 | static const struct driver_info r8153_info = { | |
207 | .description = "RTL8153 ECM Device", | |
208 | .flags = FLAG_ETHER | FLAG_POINTTOPOINT, | |
209 | .bind = r8153_bind, | |
210 | .unbind = usbnet_cdc_unbind, | |
211 | .status = usbnet_cdc_status, | |
212 | .manage_power = usbnet_manage_power, | |
213 | }; | |
214 | ||
215 | static const struct usb_device_id products[] = { | |
216 | { | |
217 | USB_DEVICE_AND_INTERFACE_INFO(REALTEK_VENDOR_ID, 0x8152, USB_CLASS_COMM, | |
218 | USB_CDC_SUBCLASS_ETHERNET, USB_CDC_PROTO_NONE), | |
219 | #if defined(CONFIG_USB_RTL8152) || defined(CONFIG_USB_RTL8152_MODULE) | |
220 | .driver_info = 0, | |
221 | #else | |
222 | .driver_info = (unsigned long) &r8152_info, | |
223 | #endif | |
224 | }, | |
225 | ||
226 | { | |
227 | USB_DEVICE_AND_INTERFACE_INFO(REALTEK_VENDOR_ID, 0x8153, USB_CLASS_COMM, | |
228 | USB_CDC_SUBCLASS_ETHERNET, USB_CDC_PROTO_NONE), | |
43779f8d | 229 | #if defined(CONFIG_USB_RTL8152) || defined(CONFIG_USB_RTL8152_MODULE) |
c073f666 | 230 | .driver_info = 0, |
231 | #else | |
232 | .driver_info = (unsigned long) &r8153_info, | |
233 | #endif | |
234 | }, | |
235 | ||
236 | { }, /* END */ | |
237 | }; | |
238 | MODULE_DEVICE_TABLE(usb, products); | |
239 | ||
240 | static struct usb_driver r815x_driver = { | |
241 | .name = "r815x", | |
242 | .id_table = products, | |
243 | .probe = usbnet_probe, | |
244 | .disconnect = usbnet_disconnect, | |
245 | .suspend = usbnet_suspend, | |
246 | .resume = usbnet_resume, | |
247 | .reset_resume = usbnet_resume, | |
248 | .supports_autosuspend = 1, | |
249 | .disable_hub_initiated_lpm = 1, | |
250 | }; | |
251 | ||
252 | module_usb_driver(r815x_driver); | |
253 | ||
254 | MODULE_AUTHOR("Hayes Wang"); | |
255 | MODULE_DESCRIPTION("Realtek USB ECM device"); | |
256 | MODULE_LICENSE("GPL"); |