blob: b2b45723e2cd64028b06f622e555f18fcac08ba4 [file] [log] [blame]
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001/* linux/arch/arm/mach-msm/devices.c
2 *
3 * Copyright (C) 2008 Google, Inc.
4 * Copyright (C) 2007-2009 HTC Corporation.
5 * Author: Thomas Tsai <thomas_tsai@htc.com>
6 *
7 * This software is licensed under the terms of the GNU General Public
8 * License version 2, as published by the Free Software Foundation, and
9 * may be copied, distributed, and modified under those terms.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 */
16
17#include <linux/kernel.h>
18#include <linux/platform_device.h>
19
20#include <linux/dma-mapping.h>
21#include <mach/msm_iomap.h>
22#include <mach/dma.h>
23#include "gpio_chip.h"
24#include "devices.h"
25#include <mach/board.h>
26#include <mach/board_htc.h>
27#include <mach/msm_hsusb.h>
28#include <linux/usb/mass_storage_function.h>
29#include <linux/usb/android.h>
30
31#include <asm/mach/flash.h>
32#include <asm/setup.h>
33#include <linux/mtd/nand.h>
34#include <linux/mtd/partitions.h>
35#include <linux/delay.h>
36#include <linux/android_pmem.h>
37#include <mach/msm_rpcrouter.h>
38#include <mach/msm_iomap.h>
39#include <asm/mach/mmc.h>
40
41static char *df_serialno = "000000000000";
42
43#if 0
44struct platform_device *devices[] __initdata = {
45 &msm_device_nand,
46 &msm_device_smd,
47 &msm_device_i2c,
48};
49
50void __init msm_add_devices(void)
51{
52 platform_add_devices(devices, ARRAY_SIZE(devices));
53}
54#endif
55
56#define HSUSB_API_INIT_PHY_PROC 2
57#define HSUSB_API_PROG 0x30000064
58#define HSUSB_API_VERS 0x10001
59static void internal_phy_reset(void)
60{
61 struct msm_rpc_endpoint *usb_ep;
62 int rc;
63 struct hsusb_phy_start_req {
64 struct rpc_request_hdr hdr;
65 } req;
66
67 printk(KERN_INFO "msm_hsusb_phy_reset\n");
68
69 usb_ep = msm_rpc_connect(HSUSB_API_PROG, HSUSB_API_VERS, 0);
70 if (IS_ERR(usb_ep)) {
71 printk(KERN_ERR "%s: init rpc failed! error: %ld\n",
72 __func__, PTR_ERR(usb_ep));
73 goto close;
74 }
75 rc = msm_rpc_call(usb_ep, HSUSB_API_INIT_PHY_PROC,
76 &req, sizeof(req), 5 * HZ);
77 if (rc < 0)
78 printk(KERN_ERR "%s: rpc call failed! (%d)\n", __func__, rc);
79
80close:
81 msm_rpc_close(usb_ep);
82}
83
84/* adjust eye diagram, disable vbusvalid interrupts */
85static int hsusb_phy_init_seq[] = { 0x40, 0x31, 0x1D, 0x0D, 0x1D, 0x10, -1 };
86
87#ifdef CONFIG_USB_FUNCTION
88static char *usb_functions[] = {
89#if defined(CONFIG_USB_FUNCTION_MASS_STORAGE) || defined(CONFIG_USB_FUNCTION_UMS)
90 "usb_mass_storage",
91#endif
92#ifdef CONFIG_USB_FUNCTION_ADB
93 "adb",
94#endif
95};
96
97static struct msm_hsusb_product usb_products[] = {
98 {
99 .product_id = 0x0c01,
100 .functions = 0x00000041, /* usb_mass_storage */
101 },
102 {
103 .product_id = 0x0c02,
104 .functions = 0x00000043, /* usb_mass_storage + adb */
105 },
106};
107#endif
108
109struct msm_hsusb_platform_data msm_hsusb_pdata = {
110 .phy_reset = internal_phy_reset,
111 .phy_init_seq = hsusb_phy_init_seq,
112#ifdef CONFIG_USB_FUNCTION
113 .vendor_id = 0x0bb4,
114 .product_id = 0x0c02,
115 .version = 0x0100,
116 .product_name = "Android Phone",
117 .manufacturer_name = "HTC",
118
119 .functions = usb_functions,
120 .num_functions = ARRAY_SIZE(usb_functions),
121 .products = usb_products,
122 .num_products = ARRAY_SIZE(usb_products),
123#endif
124};
125
126#ifdef CONFIG_USB_FUNCTION
127static struct usb_mass_storage_platform_data mass_storage_pdata = {
128 .nluns = 1,
129 .buf_size = 16384,
130 .vendor = "HTC ",
131 .product = "Android Phone ",
132 .release = 0x0100,
133};
134
135static struct platform_device usb_mass_storage_device = {
136 .name = "usb_mass_storage",
137 .id = -1,
138 .dev = {
139 .platform_data = &mass_storage_pdata,
140 },
141};
142#endif
143
144#ifdef CONFIG_USB_ANDROID
145static struct android_usb_platform_data android_usb_pdata = {
146 .vendor_id = 0x0bb4,
147 .product_id = 0x0c01,
148 .adb_product_id = 0x0c02,
149 .version = 0x0100,
150 .product_name = "Android Phone",
151 .manufacturer_name = "HTC",
152 .nluns = 1,
153};
154
155static struct platform_device android_usb_device = {
156 .name = "android_usb",
157 .id = -1,
158 .dev = {
159 .platform_data = &android_usb_pdata,
160 },
161};
162#endif
163
164void __init msm_add_usb_devices(void (*phy_reset) (void))
165{
166 /* setup */
167 if (phy_reset)
168 msm_hsusb_pdata.phy_reset = phy_reset;
169 msm_device_hsusb.dev.platform_data = &msm_hsusb_pdata;
170 platform_device_register(&msm_device_hsusb);
171#ifdef CONFIG_USB_FUNCTION_MASS_STORAGE
172 platform_device_register(&usb_mass_storage_device);
173#endif
174#ifdef CONFIG_USB_ANDROID
175 platform_device_register(&android_usb_device);
176#endif
177}
178
179static struct android_pmem_platform_data pmem_pdata = {
180 .name = "pmem",
181 .allocator_type = PMEM_ALLOCATORTYPE_ALLORNOTHING,
182 .cached = 1,
183};
184
185static struct android_pmem_platform_data pmem_adsp_pdata = {
186 .name = "pmem_adsp",
187 .allocator_type = PMEM_ALLOCATORTYPE_BUDDYBESTFIT,
188 .cached = 0,
189};
190
191static struct android_pmem_platform_data pmem_camera_pdata = {
192 .name = "pmem_camera",
193 .allocator_type = PMEM_ALLOCATORTYPE_BUDDYBESTFIT,
194 .cached = 0,
195};
196
197static struct android_pmem_platform_data pmem_gpu0_pdata = {
198 .name = "pmem_gpu0",
199 .allocator_type = PMEM_ALLOCATORTYPE_ALLORNOTHING,
200 .cached = 0,
201 .buffered = 1,
202};
203
204static struct android_pmem_platform_data pmem_gpu1_pdata = {
205 .name = "pmem_gpu1",
206 .allocator_type = PMEM_ALLOCATORTYPE_ALLORNOTHING,
207 .cached = 0,
208 .buffered = 1,
209};
210
211static struct platform_device pmem_device = {
212 .name = "android_pmem",
213 .id = 0,
214 .dev = { .platform_data = &pmem_pdata },
215};
216
217static struct platform_device pmem_adsp_device = {
218 .name = "android_pmem",
219 .id = 1,
220 .dev = { .platform_data = &pmem_adsp_pdata },
221};
222
223static struct platform_device pmem_gpu0_device = {
224 .name = "android_pmem",
225 .id = 2,
226 .dev = { .platform_data = &pmem_gpu0_pdata },
227};
228
229static struct platform_device pmem_gpu1_device = {
230 .name = "android_pmem",
231 .id = 3,
232 .dev = { .platform_data = &pmem_gpu1_pdata },
233};
234
235static struct platform_device pmem_camera_device = {
236 .name = "android_pmem",
237 .id = 4,
238 .dev = { .platform_data = &pmem_camera_pdata },
239};
240
241static struct resource ram_console_resource[] = {
242 {
243 .flags = IORESOURCE_MEM,
244 }
245};
246
247static struct platform_device ram_console_device = {
248 .name = "ram_console",
249 .id = -1,
250 .num_resources = ARRAY_SIZE(ram_console_resource),
251 .resource = ram_console_resource,
252};
253
254void __init msm_add_mem_devices(struct msm_pmem_setting *setting)
255{
256 if (setting->pmem_size) {
257 pmem_pdata.start = setting->pmem_start;
258 pmem_pdata.size = setting->pmem_size;
259 platform_device_register(&pmem_device);
260 }
261
262 if (setting->pmem_adsp_size) {
263 pmem_adsp_pdata.start = setting->pmem_adsp_start;
264 pmem_adsp_pdata.size = setting->pmem_adsp_size;
265 platform_device_register(&pmem_adsp_device);
266 }
267
268 if (setting->pmem_gpu0_size) {
269 pmem_gpu0_pdata.start = setting->pmem_gpu0_start;
270 pmem_gpu0_pdata.size = setting->pmem_gpu0_size;
271 platform_device_register(&pmem_gpu0_device);
272 }
273
274 if (setting->pmem_gpu1_size) {
275 pmem_gpu1_pdata.start = setting->pmem_gpu1_start;
276 pmem_gpu1_pdata.size = setting->pmem_gpu1_size;
277 platform_device_register(&pmem_gpu1_device);
278 }
279
280 if (setting->pmem_camera_size) {
281 pmem_camera_pdata.start = setting->pmem_camera_start;
282 pmem_camera_pdata.size = setting->pmem_camera_size;
283 platform_device_register(&pmem_camera_device);
284 }
285
286 if (setting->ram_console_size) {
287 ram_console_resource[0].start = setting->ram_console_start;
288 ram_console_resource[0].end = setting->ram_console_start
289 + setting->ram_console_size - 1;
290 platform_device_register(&ram_console_device);
291 }
292}
293
294#define PM_LIBPROG 0x30000061
295#if (CONFIG_MSM_AMSS_VERSION == 6220) || (CONFIG_MSM_AMSS_VERSION == 6225)
296#define PM_LIBVERS 0xfb837d0b
297#else
298#define PM_LIBVERS 0x10001
299#endif
300
301#if 0
302static struct platform_device *msm_serial_devices[] __initdata = {
303 &msm_device_uart1,
304 &msm_device_uart2,
305 &msm_device_uart3,
306 #ifdef CONFIG_SERIAL_MSM_HS
307 &msm_device_uart_dm1,
308 &msm_device_uart_dm2,
309 #endif
310};
311
312int __init msm_add_serial_devices(unsigned num)
313{
314 if (num > MSM_SERIAL_NUM)
315 return -EINVAL;
316
317 return platform_device_register(msm_serial_devices[num]);
318}
319#endif
320
321#define ATAG_SMI 0x4d534D71
322/* setup calls mach->fixup, then parse_tags, parse_cmdline
323 * We need to setup meminfo in mach->fixup, so this function
324 * will need to traverse each tag to find smi tag.
325 */
326int __init parse_tag_smi(const struct tag *tags)
327{
328 int smi_sz = 0, find = 0;
329 struct tag *t = (struct tag *)tags;
330
331 for (; t->hdr.size; t = tag_next(t)) {
332 if (t->hdr.tag == ATAG_SMI) {
333 printk(KERN_DEBUG "find the smi tag\n");
334 find = 1;
335 break;
336 }
337 }
338 if (!find)
339 return -1;
340
341 printk(KERN_DEBUG "parse_tag_smi: smi size = %d\n", t->u.mem.size);
342 smi_sz = t->u.mem.size;
343 return smi_sz;
344}
345__tagtable(ATAG_SMI, parse_tag_smi);
346
347
348#define ATAG_HWID 0x4d534D72
349int __init parse_tag_hwid(const struct tag *tags)
350{
351 int hwid = 0, find = 0;
352 struct tag *t = (struct tag *)tags;
353
354 for (; t->hdr.size; t = tag_next(t)) {
355 if (t->hdr.tag == ATAG_HWID) {
356 printk(KERN_DEBUG "find the hwid tag\n");
357 find = 1;
358 break;
359 }
360 }
361
362 if (find)
363 hwid = t->u.revision.rev;
364 printk(KERN_DEBUG "parse_tag_hwid: hwid = 0x%x\n", hwid);
365 return hwid;
366}
367__tagtable(ATAG_HWID, parse_tag_hwid);
368
369#define ATAG_SKUID 0x4d534D73
370int __init parse_tag_skuid(const struct tag *tags)
371{
372 int skuid = 0, find = 0;
373 struct tag *t = (struct tag *)tags;
374
375 for (; t->hdr.size; t = tag_next(t)) {
376 if (t->hdr.tag == ATAG_SKUID) {
377 printk(KERN_DEBUG "find the skuid tag\n");
378 find = 1;
379 break;
380 }
381 }
382
383 if (find)
384 skuid = t->u.revision.rev;
385 printk(KERN_DEBUG "parse_tag_skuid: hwid = 0x%x\n", skuid);
386 return skuid;
387}
388__tagtable(ATAG_SKUID, parse_tag_skuid);
389
390#define ATAG_ENGINEERID 0x4d534D75
391int __init parse_tag_engineerid(const struct tag *tags)
392{
393 int engineerid = 0, find = 0;
394 struct tag *t = (struct tag *)tags;
395
396 for (; t->hdr.size; t = tag_next(t)) {
397 if (t->hdr.tag == ATAG_ENGINEERID) {
398 printk(KERN_DEBUG "find the engineer tag\n");
399 find = 1;
400 break;
401 }
402 }
403
404 if (find)
405 engineerid = t->u.revision.rev;
406 printk(KERN_DEBUG "parse_tag_engineerid: hwid = 0x%x\n", engineerid);
407 return engineerid;
408}
409__tagtable(ATAG_ENGINEERID, parse_tag_engineerid);
410
411static int mfg_mode;
412int __init board_mfg_mode_init(char *s)
413{
414 if (!strcmp(s, "normal"))
415 mfg_mode = 0;
416 else if (!strcmp(s, "factory2"))
417 mfg_mode = 1;
418 else if (!strcmp(s, "recovery"))
419 mfg_mode = 2;
420 else if (!strcmp(s, "charge"))
421 mfg_mode = 3;
422
423 return 1;
424}
425__setup("androidboot.mode=", board_mfg_mode_init);
426
427
428int board_mfg_mode(void)
429{
430 return mfg_mode;
431}
432
433static int __init board_serialno_setup(char *serialno)
434{
435 char *str;
436
437 if (board_mfg_mode() || !strlen(serialno))
438 str = df_serialno;
439 else
440 str = serialno;
441#ifdef CONFIG_USB_FUNCTION
442 msm_hsusb_pdata.serial_number = str;
443#endif
444#ifdef CONFIG_USB_ANDROID
445 android_usb_pdata.serial_number = str;
446#endif
447 return 1;
448}
449
450__setup("androidboot.serialno=", board_serialno_setup);