blob: 3cf00b8e29b4d0db3871f587db19b98fc7448f63 [file] [log] [blame]
wdenkc6097192002-11-03 00:24:07 +00001/*
2 * (C) Copyright 2002
3 * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
wdenkdd875c72004-01-03 21:24:46 +00004 * (C) Copyright 2002
5 * Robert Schwebel, Pengutronix, <r.schwebel@pengutronix.de>
6 * (C) Copyright 2003
7 * Kai-Uwe Bloem, Auerswald GmbH & Co KG, <linux-development@auerswald.de>
wdenkc6097192002-11-03 00:24:07 +00008 *
9 * See file CREDITS for list of people who contributed to this
10 * project.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License as
14 * published by the Free Software Foundation; either version 2 of
15 * the License, or (at your option) any later version.
16 *
17 * This program is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 * GNU General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, write to the Free Software
24 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
25 * MA 02111-1307 USA
26 */
27
28/*
29 * Boot support
30 */
31#include <common.h>
32#include <command.h>
wdenkc6097192002-11-03 00:24:07 +000033#include <s_record.h>
wdenk8bde7f72003-06-27 21:31:46 +000034#include <jffs2/load_kernel.h>
wdenkc6097192002-11-03 00:24:07 +000035#include <net.h>
36
37#if (CONFIG_COMMANDS & CFG_CMD_JFFS2)
wdenkdd875c72004-01-03 21:24:46 +000038
39#include <cramfs/cramfs_fs.h>
40
wdenk180d3f72004-01-04 16:28:35 +000041extern int cramfs_check (struct part_info *info);
42extern int cramfs_load (char *loadoffset, struct part_info *info, char *filename);
43extern int cramfs_ls (struct part_info *info, char *filename);
44extern int cramfs_info (struct part_info *info);
45
wdenkc6097192002-11-03 00:24:07 +000046static int part_num=0;
47
48#ifndef CFG_JFFS_CUSTOM_PART
49
50static struct part_info part;
51
wdenk998eaae2004-04-18 19:43:36 +000052#ifndef CONFIG_JFFS2_NAND
53
wdenkc6097192002-11-03 00:24:07 +000054struct part_info*
55jffs2_part_info(int part_num)
56{
57 extern flash_info_t flash_info[]; /* info for FLASH chips */
58 int i;
59
60 if(part_num==0){
61
62 if(part.usr_priv==(void*)1)
63 return &part;
64
65 memset(&part, 0, sizeof(part));
66
67#if defined(CFG_JFFS2_FIRST_SECTOR)
68 part.offset = (unsigned char *) flash_info[CFG_JFFS2_FIRST_BANK].start[CFG_JFFS2_FIRST_SECTOR];
69#else
70 part.offset = (unsigned char *) flash_info[CFG_JFFS2_FIRST_BANK].start[0];
71#endif
72
73 /* Figure out flash partition size */
74 for (i = CFG_JFFS2_FIRST_BANK; i < CFG_JFFS2_NUM_BANKS+CFG_JFFS2_FIRST_BANK; i++)
75 part.size += flash_info[i].size;
76
77#if defined(CFG_JFFS2_FIRST_SECTOR) && (CFG_JFFS2_FIRST_SECTOR > 0)
78 part.size -=
79 flash_info[CFG_JFFS2_FIRST_BANK].start[CFG_JFFS2_FIRST_SECTOR] -
80 flash_info[CFG_JFFS2_FIRST_BANK].start[0];
81#endif
82
wdenkfc1cfcd2004-04-25 15:41:35 +000083 /* unused in current jffs2 loader */
84 part.erasesize = 0;
wdenkc6097192002-11-03 00:24:07 +000085
86 /* Mark the struct as ready */
87 part.usr_priv=(void*)1;
88
89 return &part;
90 }
91 return 0;
92}
wdenk998eaae2004-04-18 19:43:36 +000093
94#else /* CONFIG_JFFS2_NAND */
95
96struct part_info*
97jffs2_part_info(int part_num)
98{
99 if(part_num==0){
100
101 if(part.usr_priv==(void*)1)
102 return &part;
103
104 memset(&part, 0, sizeof(part));
105
106 part.offset = CONFIG_JFFS2_NAND_OFF;
107 part.size = CONFIG_JFFS2_NAND_SIZE; /* the bigger size the slower jffs2 */
108
109#ifndef CONFIG_JFFS2_NAND_DEV
110#define CONFIG_JFFS2_NAND_DEV 0
111#endif
112 /* nand device with the JFFS2 parition plus 1 */
113 part.usr_priv = (void*)(CONFIG_JFFS2_NAND_DEV+1);
114 return &part;
115 }
116 return 0;
117}
118
119#endif /* CONFIG_JFFS2_NAND */
wdenkc6097192002-11-03 00:24:07 +0000120#endif /* ifndef CFG_JFFS_CUSTOM_PART */
wdenkdd875c72004-01-03 21:24:46 +0000121
wdenkc6097192002-11-03 00:24:07 +0000122int
123do_jffs2_fsload(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
124{
wdenk8bde7f72003-06-27 21:31:46 +0000125 struct part_info* jffs2_part_info(int);
126 int jffs2_1pass_load(char *, struct part_info *,const char *);
wdenkdd875c72004-01-03 21:24:46 +0000127 char *fsname;
wdenk8bde7f72003-06-27 21:31:46 +0000128
wdenk3bac3512003-03-12 10:41:04 +0000129 char *filename = "uImage";
wdenk4b9206e2004-03-23 22:14:11 +0000130 ulong offset = load_addr;
wdenkc6097192002-11-03 00:24:07 +0000131 int size;
132 struct part_info *part;
133
134 if (argc == 2) {
135 filename = argv[1];
136 }
137 if (argc == 3) {
138 offset = simple_strtoul(argv[1], NULL, 16);
wdenk4b9206e2004-03-23 22:14:11 +0000139 load_addr = offset;
wdenkc6097192002-11-03 00:24:07 +0000140 filename = argv[2];
141 }
142
143 if (0 != (part=jffs2_part_info(part_num))){
144
wdenkdd875c72004-01-03 21:24:46 +0000145 /* check partition type for cramfs */
146 fsname = (cramfs_check(part) ? "CRAMFS" : "JFFS2");
147 printf("### %s loading '%s' to 0x%lx\n", fsname, filename, offset);
148
149 if (cramfs_check(part)) {
150 size = cramfs_load ((char *) offset, part, filename);
151 } else {
152 /* if this is not cramfs assume jffs2 */
153 size = jffs2_1pass_load((char *)offset, part, filename);
154 }
wdenkc6097192002-11-03 00:24:07 +0000155
156 if (size > 0) {
157 char buf[10];
wdenkdd875c72004-01-03 21:24:46 +0000158 printf("### %s load complete: %d bytes loaded to 0x%lx\n",
159 fsname, size, offset);
wdenkc6097192002-11-03 00:24:07 +0000160 sprintf(buf, "%x", size);
161 setenv("filesize", buf);
162 } else {
wdenkdd875c72004-01-03 21:24:46 +0000163 printf("### %s LOAD ERROR<%x> for %s!\n", fsname, size, filename);
wdenkc6097192002-11-03 00:24:07 +0000164 }
165
166 return !(size > 0);
167 }
wdenk4b9206e2004-03-23 22:14:11 +0000168 puts ("Active partition not valid\n");
wdenkc6097192002-11-03 00:24:07 +0000169 return 0;
170}
171
172int
173do_jffs2_ls(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
174{
wdenk8bde7f72003-06-27 21:31:46 +0000175 struct part_info* jffs2_part_info(int);
176 int jffs2_1pass_ls(struct part_info *,char *);
177
178 char *filename = "/";
wdenkc6097192002-11-03 00:24:07 +0000179 int ret;
180 struct part_info *part;
181
182 if (argc == 2)
183 filename = argv[1];
184
185 if (0 != (part=jffs2_part_info(part_num))){
186
wdenkdd875c72004-01-03 21:24:46 +0000187 /* check partition type for cramfs */
188 if (cramfs_check(part)) {
189 ret = cramfs_ls (part, filename);
190 } else {
191 /* if this is not cramfs assume jffs2 */
192 ret = jffs2_1pass_ls(part, filename);
193 }
wdenkc6097192002-11-03 00:24:07 +0000194
195 return (ret == 1);
196 }
wdenk4b9206e2004-03-23 22:14:11 +0000197 puts ("Active partition not valid\n");
wdenkc6097192002-11-03 00:24:07 +0000198 return 0;
199}
200
201int
202do_jffs2_fsinfo(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
203{
wdenkdd875c72004-01-03 21:24:46 +0000204 struct part_info* jffs2_part_info(int);
205 int jffs2_1pass_info(struct part_info *);
wdenkc6097192002-11-03 00:24:07 +0000206 struct part_info *part;
wdenkdd875c72004-01-03 21:24:46 +0000207 char *fsname;
208 int ret;
wdenkc6097192002-11-03 00:24:07 +0000209
wdenkdd875c72004-01-03 21:24:46 +0000210 if ((part=jffs2_part_info(part_num))){
wdenkc6097192002-11-03 00:24:07 +0000211
wdenkdd875c72004-01-03 21:24:46 +0000212 /* check partition type for cramfs */
213 fsname = (cramfs_check(part) ? "CRAMFS" : "JFFS2");
214 printf("### filesystem type is %s\n", fsname);
215
216 if (cramfs_check(part)) {
217 ret = cramfs_info (part);
218 } else {
219 /* if this is not cramfs assume jffs2 */
220 ret = jffs2_1pass_info(part);
221 }
wdenkc6097192002-11-03 00:24:07 +0000222
223 return (ret == 1);
224 }
wdenk4b9206e2004-03-23 22:14:11 +0000225 puts ("Active partition not valid\n");
wdenkc6097192002-11-03 00:24:07 +0000226 return 0;
227}
228
229int
230do_jffs2_chpart(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
231{
232 int tmp_part;
wdenkdd875c72004-01-03 21:24:46 +0000233 char str_part_num[3];
wdenk8bde7f72003-06-27 21:31:46 +0000234 struct part_info* jffs2_part_info(int);
wdenkc6097192002-11-03 00:24:07 +0000235
wdenk8bde7f72003-06-27 21:31:46 +0000236 if (argc >= 2) {
wdenkc6097192002-11-03 00:24:07 +0000237 tmp_part = simple_strtoul(argv[1], NULL, 16);
238 }else{
wdenk4b9206e2004-03-23 22:14:11 +0000239 puts ("Need partition number in argument list\n");
wdenkc6097192002-11-03 00:24:07 +0000240 return 0;
241
242 }
243
244 if (jffs2_part_info(tmp_part)){
wdenk7205e402003-09-10 22:30:53 +0000245 printf("Partition changed to %d\n",tmp_part);
wdenkc6097192002-11-03 00:24:07 +0000246 part_num=tmp_part;
wdenkdd875c72004-01-03 21:24:46 +0000247 sprintf(str_part_num, "%d", part_num);
248 setenv("partition", str_part_num);
wdenkc6097192002-11-03 00:24:07 +0000249 return 0;
250 }
251
252 printf("Partition %d is not valid partiton\n",tmp_part);
253 return 0;
254
255}
wdenk8bde7f72003-06-27 21:31:46 +0000256
257/***************************************************/
258
wdenk0d498392003-07-01 21:06:45 +0000259U_BOOT_CMD(
260 fsload, 3, 0, do_jffs2_fsload,
wdenk8bde7f72003-06-27 21:31:46 +0000261 "fsload - load binary file from a filesystem image\n",
262 "[ off ] [ filename ]\n"
263 " - load binary file from flash bank\n"
264 " with offset 'off'\n"
265);
266
wdenk0d498392003-07-01 21:06:45 +0000267U_BOOT_CMD(
268 fsinfo, 1, 1, do_jffs2_fsinfo,
wdenk8bde7f72003-06-27 21:31:46 +0000269 "fsinfo - print information about filesystems\n",
270 " - print information about filesystems\n"
271);
272
wdenk0d498392003-07-01 21:06:45 +0000273U_BOOT_CMD(
274 ls, 2, 1, do_jffs2_ls,
wdenk8bde7f72003-06-27 21:31:46 +0000275 "ls - list files in a directory (default /)\n",
276 "[ directory ]\n"
277 " - list files in a directory.\n"
278);
279
wdenk0d498392003-07-01 21:06:45 +0000280U_BOOT_CMD(
281 chpart, 2, 0, do_jffs2_chpart,
wdenk8bde7f72003-06-27 21:31:46 +0000282 "chpart - change active partition\n",
283 " - change active partition\n"
284);
285
wdenkc6097192002-11-03 00:24:07 +0000286#endif /* CFG_CMD_JFFS2 */