recovery: restructure the recovery source tree, add new files
[kernel/u-boot.git] / recovery / board / samsung / universal / universal.c
1 /*\r
2  * (C) Copyright 2010 Samsung Electronics\r
3  *\r
4  * See file CREDITS for list of people who contributed to this\r
5  * project.\r
6  *\r
7  * This program is free software; you can redistribute it and/or\r
8  * modify it under the terms of the GNU General Public License as\r
9  * published by the Free Software Foundation; either version 2 of\r
10  * the License, or (at your option) any later version.\r
11  *\r
12  * This program is distributed in the hope that it will be useful,\r
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of\r
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\r
15  * GNU General Public License for more details.\r
16  *\r
17  * You should have received a copy of the GNU General Public License\r
18  * along with this program; if not, write to the Free Software\r
19  * Foundation, Inc., 59 Temple Place, Suite 330, Boston,\r
20  * MA 02111-1307 USA\r
21  */\r
22 \r
23 #include <common.h>\r
24 #include <linux/mtd/compat.h>\r
25 #include <linux/mtd/mtd.h>\r
26 #include <linux/mtd/onenand.h>\r
27 #include <asm/io.h>\r
28 #include <asm/arch/gpio.h>\r
29 #include <asm/arch/keypad.h>\r
30 #include "../../../onenand.h"\r
31 \r
32 typedef int (init_fnc_t) (void);\r
33 \r
34 DECLARE_GLOBAL_DATA_PTR;\r
35 \r
36 static struct s5pc110_gpio *s5pc110_gpio;\r
37 \r
38 static void sdelay(unsigned long usec)\r
39 {\r
40         ulong kv;\r
41 \r
42         do {\r
43                 kv = 0x100000;\r
44                 while (kv-- > 0) {\r
45                 }\r
46                 usec--;\r
47         } while(usec);\r
48 }\r
49 \r
50 static int check_keypad(void)\r
51 {\r
52         unsigned int condition = 0;\r
53         unsigned int reg, value;\r
54         unsigned int col_num, row_num;\r
55         unsigned int col_mask;\r
56         unsigned int col_mask_shift;\r
57         unsigned int row_state[4] = {0, 0, 0, 0};\r
58         unsigned int i;\r
59 \r
60         /* board is limo universal */\r
61 \r
62         row_num = 2;\r
63         col_num = 3;\r
64 \r
65         for (i = 0; i < sizeof(row_state)/sizeof(int); i++) {\r
66                 row_state[i] = 0;\r
67         }\r
68         \r
69         for (i = 0; i < row_num; i++) {\r
70                 /* Set GPH3[3:0] to KP_ROW[3:0] */\r
71                 gpio_cfg_pin(&s5pc110_gpio->gpio_h3, i, 0x3);\r
72                 gpio_set_pull(&s5pc110_gpio->gpio_h3, i, GPIO_PULL_UP);\r
73         }\r
74 \r
75         for (i = 0; i < col_num; i++)\r
76                 /* Set GPH2[3:0] to KP_COL[3:0] */\r
77                 gpio_cfg_pin(&s5pc110_gpio->gpio_h2, i, 0x3);\r
78 \r
79         reg = S5PC110_KEYPAD_BASE;\r
80         col_mask = S5PC110_KEYIFCOLEN_MASK;\r
81         col_mask_shift = 8;\r
82 \r
83         /* KEYIFCOL reg clear */\r
84         writel(0, reg + S5PC1XX_KEYIFCOL_OFFSET);\r
85 \r
86         /* key_scan */\r
87         for (i = 0; i < col_num; i++) {\r
88                 value = col_mask;\r
89                 value &= ~(1 << i) << col_mask_shift;\r
90 \r
91                 writel(value, reg + S5PC1XX_KEYIFCOL_OFFSET);\r
92                 sdelay(1000);\r
93 \r
94                 value = readl(reg + S5PC1XX_KEYIFROW_OFFSET);\r
95                 row_state[i] = ~value & ((1 << row_num) - 1);\r
96         }\r
97 \r
98         /* KEYIFCOL reg clear */\r
99         writel(0, reg + S5PC1XX_KEYIFCOL_OFFSET);\r
100 \r
101         /* check volume up/down */\r
102         if ((row_state[1] & 0x3) == 0x3) {\r
103                 condition = 1;\r
104         }\r
105 \r
106         return condition;\r
107 }\r
108 \r
109 static int check_block(void)\r
110 {\r
111         struct onenand_op *onenand_ops = onenand_get_interface();\r
112         struct mtd_info *mtd = onenand_ops->mtd;\r
113         struct onenand_chip *this = mtd->priv;\r
114         int i;\r
115         int page_to_check = 4;\r
116         int ret, retlen = 0;\r
117         ulong blocksize = 1 << this->erase_shift;\r
118         ulong pagesize = 1 << this->page_shift;\r
119         u_char *down_ram_addr = (unsigned char *)CONFIG_SYS_DOWN_ADDR;\r
120         ulong uboot_start_addr = CONFIG_RECOVERY_BOOTSTART_BLOCK << this->erase_shift;\r
121         u_char verify_buf[0x10];\r
122 \r
123         onenand_ops->read(uboot_start_addr, blocksize, &retlen, down_ram_addr, 0);\r
124         if (retlen != blocksize)\r
125         {\r
126                 return 1;\r
127         }\r
128 \r
129         memset(verify_buf, 0xFF, sizeof(verify_buf));\r
130 \r
131         for (i = 0; i < page_to_check; i++) {\r
132                 ret = memcmp(down_ram_addr + pagesize*i, verify_buf, sizeof(verify_buf));\r
133                 if (ret)\r
134                 {\r
135                         break;\r
136                 }\r
137         }\r
138 \r
139         if (i == page_to_check) {\r
140                 return 1;\r
141         }\r
142         \r
143         return 0;\r
144 }\r
145 \r
146 int board_check_condition(void)\r
147 {\r
148         if (check_keypad()) {\r
149                 return 1;\r
150         }\r
151         \r
152         if (check_block()) {\r
153                 return 1;\r
154         }\r
155 \r
156         return 0;\r
157 }\r
158 \r
159 int board_load_uboot(unsigned char *buf)\r
160 {\r
161         struct mtd_info *mtd = &onenand_mtd;\r
162         struct onenand_chip *this = mtd->priv;\r
163         int offset;\r
164         size_t size;\r
165         size_t ret;\r
166 \r
167         offset = CONFIG_RECOVERY_BOOTSTART_BLOCK << this->erase_shift;\r
168         size = CONFIG_SYS_MONITOR_LEN;\r
169 \r
170         mtd->read(mtd, offset, size, &ret, buf);\r
171 \r
172         if (size != ret) {\r
173                 return -1;\r
174         }\r
175                 \r
176         return 0;\r
177 }\r
178 \r
179 int board_update_uboot(void)\r
180 {\r
181         return 0;\r
182 }\r
183 \r
184 void board_recovery_init(void)\r
185 {\r
186         /* Set Initial global variables */\r
187         s5pc110_gpio = (struct s5pc110_gpio *) S5PC110_GPIO_BASE;\r
188 }\r
189 \r