xref: /OK3568_Linux_fs/u-boot/cmd/bootmenu.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  * (C) Copyright 2011-2013 Pali Rohár <pali.rohar@gmail.com>
3*4882a593Smuzhiyun  *
4*4882a593Smuzhiyun  * SPDX-License-Identifier:	GPL-2.0+
5*4882a593Smuzhiyun  */
6*4882a593Smuzhiyun 
7*4882a593Smuzhiyun #include <common.h>
8*4882a593Smuzhiyun #include <command.h>
9*4882a593Smuzhiyun #include <ansi.h>
10*4882a593Smuzhiyun #include <menu.h>
11*4882a593Smuzhiyun #include <watchdog.h>
12*4882a593Smuzhiyun #include <malloc.h>
13*4882a593Smuzhiyun #include <linux/string.h>
14*4882a593Smuzhiyun 
15*4882a593Smuzhiyun /* maximum bootmenu entries */
16*4882a593Smuzhiyun #define MAX_COUNT	99
17*4882a593Smuzhiyun 
18*4882a593Smuzhiyun /* maximal size of bootmenu env
19*4882a593Smuzhiyun  *  9 = strlen("bootmenu_")
20*4882a593Smuzhiyun  *  2 = strlen(MAX_COUNT)
21*4882a593Smuzhiyun  *  1 = NULL term
22*4882a593Smuzhiyun  */
23*4882a593Smuzhiyun #define MAX_ENV_SIZE	(9 + 2 + 1)
24*4882a593Smuzhiyun 
25*4882a593Smuzhiyun struct bootmenu_entry {
26*4882a593Smuzhiyun 	unsigned short int num;		/* unique number 0 .. MAX_COUNT */
27*4882a593Smuzhiyun 	char key[3];			/* key identifier of number */
28*4882a593Smuzhiyun 	char *title;			/* title of entry */
29*4882a593Smuzhiyun 	char *command;			/* hush command of entry */
30*4882a593Smuzhiyun 	struct bootmenu_data *menu;	/* this bootmenu */
31*4882a593Smuzhiyun 	struct bootmenu_entry *next;	/* next menu entry (num+1) */
32*4882a593Smuzhiyun };
33*4882a593Smuzhiyun 
34*4882a593Smuzhiyun struct bootmenu_data {
35*4882a593Smuzhiyun 	int delay;			/* delay for autoboot */
36*4882a593Smuzhiyun 	int active;			/* active menu entry */
37*4882a593Smuzhiyun 	int count;			/* total count of menu entries */
38*4882a593Smuzhiyun 	struct bootmenu_entry *first;	/* first menu entry */
39*4882a593Smuzhiyun };
40*4882a593Smuzhiyun 
41*4882a593Smuzhiyun enum bootmenu_key {
42*4882a593Smuzhiyun 	KEY_NONE = 0,
43*4882a593Smuzhiyun 	KEY_UP,
44*4882a593Smuzhiyun 	KEY_DOWN,
45*4882a593Smuzhiyun 	KEY_SELECT,
46*4882a593Smuzhiyun };
47*4882a593Smuzhiyun 
bootmenu_getoption(unsigned short int n)48*4882a593Smuzhiyun static char *bootmenu_getoption(unsigned short int n)
49*4882a593Smuzhiyun {
50*4882a593Smuzhiyun 	char name[MAX_ENV_SIZE];
51*4882a593Smuzhiyun 
52*4882a593Smuzhiyun 	if (n > MAX_COUNT)
53*4882a593Smuzhiyun 		return NULL;
54*4882a593Smuzhiyun 
55*4882a593Smuzhiyun 	sprintf(name, "bootmenu_%d", n);
56*4882a593Smuzhiyun 	return env_get(name);
57*4882a593Smuzhiyun }
58*4882a593Smuzhiyun 
bootmenu_print_entry(void * data)59*4882a593Smuzhiyun static void bootmenu_print_entry(void *data)
60*4882a593Smuzhiyun {
61*4882a593Smuzhiyun 	struct bootmenu_entry *entry = data;
62*4882a593Smuzhiyun 	int reverse = (entry->menu->active == entry->num);
63*4882a593Smuzhiyun 
64*4882a593Smuzhiyun 	/*
65*4882a593Smuzhiyun 	 * Move cursor to line where the entry will be drown (entry->num)
66*4882a593Smuzhiyun 	 * First 3 lines contain bootmenu header + 1 empty line
67*4882a593Smuzhiyun 	 */
68*4882a593Smuzhiyun 	printf(ANSI_CURSOR_POSITION, entry->num + 4, 1);
69*4882a593Smuzhiyun 
70*4882a593Smuzhiyun 	puts("     ");
71*4882a593Smuzhiyun 
72*4882a593Smuzhiyun 	if (reverse)
73*4882a593Smuzhiyun 		puts(ANSI_COLOR_REVERSE);
74*4882a593Smuzhiyun 
75*4882a593Smuzhiyun 	puts(entry->title);
76*4882a593Smuzhiyun 
77*4882a593Smuzhiyun 	if (reverse)
78*4882a593Smuzhiyun 		puts(ANSI_COLOR_RESET);
79*4882a593Smuzhiyun }
80*4882a593Smuzhiyun 
bootmenu_autoboot_loop(struct bootmenu_data * menu,enum bootmenu_key * key,int * esc)81*4882a593Smuzhiyun static void bootmenu_autoboot_loop(struct bootmenu_data *menu,
82*4882a593Smuzhiyun 				enum bootmenu_key *key, int *esc)
83*4882a593Smuzhiyun {
84*4882a593Smuzhiyun 	int i, c;
85*4882a593Smuzhiyun 
86*4882a593Smuzhiyun 	if (menu->delay > 0) {
87*4882a593Smuzhiyun 		printf(ANSI_CURSOR_POSITION, menu->count + 5, 1);
88*4882a593Smuzhiyun 		printf("  Hit any key to stop autoboot: %2d ", menu->delay);
89*4882a593Smuzhiyun 	}
90*4882a593Smuzhiyun 
91*4882a593Smuzhiyun 	while (menu->delay > 0) {
92*4882a593Smuzhiyun 		for (i = 0; i < 100; ++i) {
93*4882a593Smuzhiyun 			if (!tstc()) {
94*4882a593Smuzhiyun 				WATCHDOG_RESET();
95*4882a593Smuzhiyun 				mdelay(10);
96*4882a593Smuzhiyun 				continue;
97*4882a593Smuzhiyun 			}
98*4882a593Smuzhiyun 
99*4882a593Smuzhiyun 			menu->delay = -1;
100*4882a593Smuzhiyun 			c = getc();
101*4882a593Smuzhiyun 
102*4882a593Smuzhiyun 			switch (c) {
103*4882a593Smuzhiyun 			case '\e':
104*4882a593Smuzhiyun 				*esc = 1;
105*4882a593Smuzhiyun 				*key = KEY_NONE;
106*4882a593Smuzhiyun 				break;
107*4882a593Smuzhiyun 			case '\r':
108*4882a593Smuzhiyun 				*key = KEY_SELECT;
109*4882a593Smuzhiyun 				break;
110*4882a593Smuzhiyun 			default:
111*4882a593Smuzhiyun 				*key = KEY_NONE;
112*4882a593Smuzhiyun 				break;
113*4882a593Smuzhiyun 			}
114*4882a593Smuzhiyun 
115*4882a593Smuzhiyun 			break;
116*4882a593Smuzhiyun 		}
117*4882a593Smuzhiyun 
118*4882a593Smuzhiyun 		if (menu->delay < 0)
119*4882a593Smuzhiyun 			break;
120*4882a593Smuzhiyun 
121*4882a593Smuzhiyun 		--menu->delay;
122*4882a593Smuzhiyun 		printf("\b\b\b%2d ", menu->delay);
123*4882a593Smuzhiyun 	}
124*4882a593Smuzhiyun 
125*4882a593Smuzhiyun 	printf(ANSI_CURSOR_POSITION, menu->count + 5, 1);
126*4882a593Smuzhiyun 	puts(ANSI_CLEAR_LINE);
127*4882a593Smuzhiyun 
128*4882a593Smuzhiyun 	if (menu->delay == 0)
129*4882a593Smuzhiyun 		*key = KEY_SELECT;
130*4882a593Smuzhiyun }
131*4882a593Smuzhiyun 
bootmenu_loop(struct bootmenu_data * menu,enum bootmenu_key * key,int * esc)132*4882a593Smuzhiyun static void bootmenu_loop(struct bootmenu_data *menu,
133*4882a593Smuzhiyun 		enum bootmenu_key *key, int *esc)
134*4882a593Smuzhiyun {
135*4882a593Smuzhiyun 	int c;
136*4882a593Smuzhiyun 
137*4882a593Smuzhiyun 	while (!tstc()) {
138*4882a593Smuzhiyun 		WATCHDOG_RESET();
139*4882a593Smuzhiyun 		mdelay(10);
140*4882a593Smuzhiyun 	}
141*4882a593Smuzhiyun 
142*4882a593Smuzhiyun 	c = getc();
143*4882a593Smuzhiyun 
144*4882a593Smuzhiyun 	switch (*esc) {
145*4882a593Smuzhiyun 	case 0:
146*4882a593Smuzhiyun 		/* First char of ANSI escape sequence '\e' */
147*4882a593Smuzhiyun 		if (c == '\e') {
148*4882a593Smuzhiyun 			*esc = 1;
149*4882a593Smuzhiyun 			*key = KEY_NONE;
150*4882a593Smuzhiyun 		}
151*4882a593Smuzhiyun 		break;
152*4882a593Smuzhiyun 	case 1:
153*4882a593Smuzhiyun 		/* Second char of ANSI '[' */
154*4882a593Smuzhiyun 		if (c == '[') {
155*4882a593Smuzhiyun 			*esc = 2;
156*4882a593Smuzhiyun 			*key = KEY_NONE;
157*4882a593Smuzhiyun 		} else {
158*4882a593Smuzhiyun 			*esc = 0;
159*4882a593Smuzhiyun 		}
160*4882a593Smuzhiyun 		break;
161*4882a593Smuzhiyun 	case 2:
162*4882a593Smuzhiyun 	case 3:
163*4882a593Smuzhiyun 		/* Third char of ANSI (number '1') - optional */
164*4882a593Smuzhiyun 		if (*esc == 2 && c == '1') {
165*4882a593Smuzhiyun 			*esc = 3;
166*4882a593Smuzhiyun 			*key = KEY_NONE;
167*4882a593Smuzhiyun 			break;
168*4882a593Smuzhiyun 		}
169*4882a593Smuzhiyun 
170*4882a593Smuzhiyun 		*esc = 0;
171*4882a593Smuzhiyun 
172*4882a593Smuzhiyun 		/* ANSI 'A' - key up was pressed */
173*4882a593Smuzhiyun 		if (c == 'A')
174*4882a593Smuzhiyun 			*key = KEY_UP;
175*4882a593Smuzhiyun 		/* ANSI 'B' - key down was pressed */
176*4882a593Smuzhiyun 		else if (c == 'B')
177*4882a593Smuzhiyun 			*key = KEY_DOWN;
178*4882a593Smuzhiyun 		/* other key was pressed */
179*4882a593Smuzhiyun 		else
180*4882a593Smuzhiyun 			*key = KEY_NONE;
181*4882a593Smuzhiyun 
182*4882a593Smuzhiyun 		break;
183*4882a593Smuzhiyun 	}
184*4882a593Smuzhiyun 
185*4882a593Smuzhiyun 	/* enter key was pressed */
186*4882a593Smuzhiyun 	if (c == '\r')
187*4882a593Smuzhiyun 		*key = KEY_SELECT;
188*4882a593Smuzhiyun }
189*4882a593Smuzhiyun 
bootmenu_choice_entry(void * data)190*4882a593Smuzhiyun static char *bootmenu_choice_entry(void *data)
191*4882a593Smuzhiyun {
192*4882a593Smuzhiyun 	struct bootmenu_data *menu = data;
193*4882a593Smuzhiyun 	struct bootmenu_entry *iter;
194*4882a593Smuzhiyun 	enum bootmenu_key key = KEY_NONE;
195*4882a593Smuzhiyun 	int esc = 0;
196*4882a593Smuzhiyun 	int i;
197*4882a593Smuzhiyun 
198*4882a593Smuzhiyun 	while (1) {
199*4882a593Smuzhiyun 		if (menu->delay >= 0) {
200*4882a593Smuzhiyun 			/* Autoboot was not stopped */
201*4882a593Smuzhiyun 			bootmenu_autoboot_loop(menu, &key, &esc);
202*4882a593Smuzhiyun 		} else {
203*4882a593Smuzhiyun 			/* Some key was pressed, so autoboot was stopped */
204*4882a593Smuzhiyun 			bootmenu_loop(menu, &key, &esc);
205*4882a593Smuzhiyun 		}
206*4882a593Smuzhiyun 
207*4882a593Smuzhiyun 		switch (key) {
208*4882a593Smuzhiyun 		case KEY_UP:
209*4882a593Smuzhiyun 			if (menu->active > 0)
210*4882a593Smuzhiyun 				--menu->active;
211*4882a593Smuzhiyun 			/* no menu key selected, regenerate menu */
212*4882a593Smuzhiyun 			return NULL;
213*4882a593Smuzhiyun 		case KEY_DOWN:
214*4882a593Smuzhiyun 			if (menu->active < menu->count - 1)
215*4882a593Smuzhiyun 				++menu->active;
216*4882a593Smuzhiyun 			/* no menu key selected, regenerate menu */
217*4882a593Smuzhiyun 			return NULL;
218*4882a593Smuzhiyun 		case KEY_SELECT:
219*4882a593Smuzhiyun 			iter = menu->first;
220*4882a593Smuzhiyun 			for (i = 0; i < menu->active; ++i)
221*4882a593Smuzhiyun 				iter = iter->next;
222*4882a593Smuzhiyun 			return iter->key;
223*4882a593Smuzhiyun 		default:
224*4882a593Smuzhiyun 			break;
225*4882a593Smuzhiyun 		}
226*4882a593Smuzhiyun 	}
227*4882a593Smuzhiyun 
228*4882a593Smuzhiyun 	/* never happens */
229*4882a593Smuzhiyun 	debug("bootmenu: this should not happen");
230*4882a593Smuzhiyun 	return NULL;
231*4882a593Smuzhiyun }
232*4882a593Smuzhiyun 
bootmenu_destroy(struct bootmenu_data * menu)233*4882a593Smuzhiyun static void bootmenu_destroy(struct bootmenu_data *menu)
234*4882a593Smuzhiyun {
235*4882a593Smuzhiyun 	struct bootmenu_entry *iter = menu->first;
236*4882a593Smuzhiyun 	struct bootmenu_entry *next;
237*4882a593Smuzhiyun 
238*4882a593Smuzhiyun 	while (iter) {
239*4882a593Smuzhiyun 		next = iter->next;
240*4882a593Smuzhiyun 		free(iter->title);
241*4882a593Smuzhiyun 		free(iter->command);
242*4882a593Smuzhiyun 		free(iter);
243*4882a593Smuzhiyun 		iter = next;
244*4882a593Smuzhiyun 	}
245*4882a593Smuzhiyun 	free(menu);
246*4882a593Smuzhiyun }
247*4882a593Smuzhiyun 
bootmenu_create(int delay)248*4882a593Smuzhiyun static struct bootmenu_data *bootmenu_create(int delay)
249*4882a593Smuzhiyun {
250*4882a593Smuzhiyun 	unsigned short int i = 0;
251*4882a593Smuzhiyun 	const char *option;
252*4882a593Smuzhiyun 	struct bootmenu_data *menu;
253*4882a593Smuzhiyun 	struct bootmenu_entry *iter = NULL;
254*4882a593Smuzhiyun 
255*4882a593Smuzhiyun 	int len;
256*4882a593Smuzhiyun 	char *sep;
257*4882a593Smuzhiyun 	struct bootmenu_entry *entry;
258*4882a593Smuzhiyun 
259*4882a593Smuzhiyun 	menu = malloc(sizeof(struct bootmenu_data));
260*4882a593Smuzhiyun 	if (!menu)
261*4882a593Smuzhiyun 		return NULL;
262*4882a593Smuzhiyun 
263*4882a593Smuzhiyun 	menu->delay = delay;
264*4882a593Smuzhiyun 	menu->active = 0;
265*4882a593Smuzhiyun 	menu->first = NULL;
266*4882a593Smuzhiyun 
267*4882a593Smuzhiyun 	while ((option = bootmenu_getoption(i))) {
268*4882a593Smuzhiyun 		sep = strchr(option, '=');
269*4882a593Smuzhiyun 		if (!sep) {
270*4882a593Smuzhiyun 			printf("Invalid bootmenu entry: %s\n", option);
271*4882a593Smuzhiyun 			break;
272*4882a593Smuzhiyun 		}
273*4882a593Smuzhiyun 
274*4882a593Smuzhiyun 		entry = malloc(sizeof(struct bootmenu_entry));
275*4882a593Smuzhiyun 		if (!entry)
276*4882a593Smuzhiyun 			goto cleanup;
277*4882a593Smuzhiyun 
278*4882a593Smuzhiyun 		len = sep-option;
279*4882a593Smuzhiyun 		entry->title = malloc(len + 1);
280*4882a593Smuzhiyun 		if (!entry->title) {
281*4882a593Smuzhiyun 			free(entry);
282*4882a593Smuzhiyun 			goto cleanup;
283*4882a593Smuzhiyun 		}
284*4882a593Smuzhiyun 		memcpy(entry->title, option, len);
285*4882a593Smuzhiyun 		entry->title[len] = 0;
286*4882a593Smuzhiyun 
287*4882a593Smuzhiyun 		len = strlen(sep + 1);
288*4882a593Smuzhiyun 		entry->command = malloc(len + 1);
289*4882a593Smuzhiyun 		if (!entry->command) {
290*4882a593Smuzhiyun 			free(entry->title);
291*4882a593Smuzhiyun 			free(entry);
292*4882a593Smuzhiyun 			goto cleanup;
293*4882a593Smuzhiyun 		}
294*4882a593Smuzhiyun 		memcpy(entry->command, sep + 1, len);
295*4882a593Smuzhiyun 		entry->command[len] = 0;
296*4882a593Smuzhiyun 
297*4882a593Smuzhiyun 		sprintf(entry->key, "%d", i);
298*4882a593Smuzhiyun 
299*4882a593Smuzhiyun 		entry->num = i;
300*4882a593Smuzhiyun 		entry->menu = menu;
301*4882a593Smuzhiyun 		entry->next = NULL;
302*4882a593Smuzhiyun 
303*4882a593Smuzhiyun 		if (!iter)
304*4882a593Smuzhiyun 			menu->first = entry;
305*4882a593Smuzhiyun 		else
306*4882a593Smuzhiyun 			iter->next = entry;
307*4882a593Smuzhiyun 
308*4882a593Smuzhiyun 		iter = entry;
309*4882a593Smuzhiyun 		++i;
310*4882a593Smuzhiyun 
311*4882a593Smuzhiyun 		if (i == MAX_COUNT - 1)
312*4882a593Smuzhiyun 			break;
313*4882a593Smuzhiyun 	}
314*4882a593Smuzhiyun 
315*4882a593Smuzhiyun 	/* Add U-Boot console entry at the end */
316*4882a593Smuzhiyun 	if (i <= MAX_COUNT - 1) {
317*4882a593Smuzhiyun 		entry = malloc(sizeof(struct bootmenu_entry));
318*4882a593Smuzhiyun 		if (!entry)
319*4882a593Smuzhiyun 			goto cleanup;
320*4882a593Smuzhiyun 
321*4882a593Smuzhiyun 		entry->title = strdup("U-Boot console");
322*4882a593Smuzhiyun 		if (!entry->title) {
323*4882a593Smuzhiyun 			free(entry);
324*4882a593Smuzhiyun 			goto cleanup;
325*4882a593Smuzhiyun 		}
326*4882a593Smuzhiyun 
327*4882a593Smuzhiyun 		entry->command = strdup("");
328*4882a593Smuzhiyun 		if (!entry->command) {
329*4882a593Smuzhiyun 			free(entry->title);
330*4882a593Smuzhiyun 			free(entry);
331*4882a593Smuzhiyun 			goto cleanup;
332*4882a593Smuzhiyun 		}
333*4882a593Smuzhiyun 
334*4882a593Smuzhiyun 		sprintf(entry->key, "%d", i);
335*4882a593Smuzhiyun 
336*4882a593Smuzhiyun 		entry->num = i;
337*4882a593Smuzhiyun 		entry->menu = menu;
338*4882a593Smuzhiyun 		entry->next = NULL;
339*4882a593Smuzhiyun 
340*4882a593Smuzhiyun 		if (!iter)
341*4882a593Smuzhiyun 			menu->first = entry;
342*4882a593Smuzhiyun 		else
343*4882a593Smuzhiyun 			iter->next = entry;
344*4882a593Smuzhiyun 
345*4882a593Smuzhiyun 		iter = entry;
346*4882a593Smuzhiyun 		++i;
347*4882a593Smuzhiyun 	}
348*4882a593Smuzhiyun 
349*4882a593Smuzhiyun 	menu->count = i;
350*4882a593Smuzhiyun 	return menu;
351*4882a593Smuzhiyun 
352*4882a593Smuzhiyun cleanup:
353*4882a593Smuzhiyun 	bootmenu_destroy(menu);
354*4882a593Smuzhiyun 	return NULL;
355*4882a593Smuzhiyun }
356*4882a593Smuzhiyun 
bootmenu_show(int delay)357*4882a593Smuzhiyun static void bootmenu_show(int delay)
358*4882a593Smuzhiyun {
359*4882a593Smuzhiyun 	int init = 0;
360*4882a593Smuzhiyun 	void *choice = NULL;
361*4882a593Smuzhiyun 	char *title = NULL;
362*4882a593Smuzhiyun 	char *command = NULL;
363*4882a593Smuzhiyun 	struct menu *menu;
364*4882a593Smuzhiyun 	struct bootmenu_data *bootmenu;
365*4882a593Smuzhiyun 	struct bootmenu_entry *iter;
366*4882a593Smuzhiyun 	char *option, *sep;
367*4882a593Smuzhiyun 
368*4882a593Smuzhiyun 	/* If delay is 0 do not create menu, just run first entry */
369*4882a593Smuzhiyun 	if (delay == 0) {
370*4882a593Smuzhiyun 		option = bootmenu_getoption(0);
371*4882a593Smuzhiyun 		if (!option) {
372*4882a593Smuzhiyun 			puts("bootmenu option 0 was not found\n");
373*4882a593Smuzhiyun 			return;
374*4882a593Smuzhiyun 		}
375*4882a593Smuzhiyun 		sep = strchr(option, '=');
376*4882a593Smuzhiyun 		if (!sep) {
377*4882a593Smuzhiyun 			puts("bootmenu option 0 is invalid\n");
378*4882a593Smuzhiyun 			return;
379*4882a593Smuzhiyun 		}
380*4882a593Smuzhiyun 		run_command(sep+1, 0);
381*4882a593Smuzhiyun 		return;
382*4882a593Smuzhiyun 	}
383*4882a593Smuzhiyun 
384*4882a593Smuzhiyun 	bootmenu = bootmenu_create(delay);
385*4882a593Smuzhiyun 	if (!bootmenu)
386*4882a593Smuzhiyun 		return;
387*4882a593Smuzhiyun 
388*4882a593Smuzhiyun 	menu = menu_create(NULL, bootmenu->delay, 1, bootmenu_print_entry,
389*4882a593Smuzhiyun 			   bootmenu_choice_entry, bootmenu);
390*4882a593Smuzhiyun 	if (!menu) {
391*4882a593Smuzhiyun 		bootmenu_destroy(bootmenu);
392*4882a593Smuzhiyun 		return;
393*4882a593Smuzhiyun 	}
394*4882a593Smuzhiyun 
395*4882a593Smuzhiyun 	for (iter = bootmenu->first; iter; iter = iter->next) {
396*4882a593Smuzhiyun 		if (!menu_item_add(menu, iter->key, iter))
397*4882a593Smuzhiyun 			goto cleanup;
398*4882a593Smuzhiyun 	}
399*4882a593Smuzhiyun 
400*4882a593Smuzhiyun 	/* Default menu entry is always first */
401*4882a593Smuzhiyun 	menu_default_set(menu, "0");
402*4882a593Smuzhiyun 
403*4882a593Smuzhiyun 	puts(ANSI_CURSOR_HIDE);
404*4882a593Smuzhiyun 	puts(ANSI_CLEAR_CONSOLE);
405*4882a593Smuzhiyun 	printf(ANSI_CURSOR_POSITION, 1, 1);
406*4882a593Smuzhiyun 
407*4882a593Smuzhiyun 	init = 1;
408*4882a593Smuzhiyun 
409*4882a593Smuzhiyun 	if (menu_get_choice(menu, &choice)) {
410*4882a593Smuzhiyun 		iter = choice;
411*4882a593Smuzhiyun 		title = strdup(iter->title);
412*4882a593Smuzhiyun 		command = strdup(iter->command);
413*4882a593Smuzhiyun 	}
414*4882a593Smuzhiyun 
415*4882a593Smuzhiyun cleanup:
416*4882a593Smuzhiyun 	menu_destroy(menu);
417*4882a593Smuzhiyun 	bootmenu_destroy(bootmenu);
418*4882a593Smuzhiyun 
419*4882a593Smuzhiyun 	if (init) {
420*4882a593Smuzhiyun 		puts(ANSI_CURSOR_SHOW);
421*4882a593Smuzhiyun 		puts(ANSI_CLEAR_CONSOLE);
422*4882a593Smuzhiyun 		printf(ANSI_CURSOR_POSITION, 1, 1);
423*4882a593Smuzhiyun 	}
424*4882a593Smuzhiyun 
425*4882a593Smuzhiyun 	if (title && command) {
426*4882a593Smuzhiyun 		debug("Starting entry '%s'\n", title);
427*4882a593Smuzhiyun 		free(title);
428*4882a593Smuzhiyun 		run_command(command, 0);
429*4882a593Smuzhiyun 		free(command);
430*4882a593Smuzhiyun 	}
431*4882a593Smuzhiyun 
432*4882a593Smuzhiyun #ifdef CONFIG_POSTBOOTMENU
433*4882a593Smuzhiyun 	run_command(CONFIG_POSTBOOTMENU, 0);
434*4882a593Smuzhiyun #endif
435*4882a593Smuzhiyun }
436*4882a593Smuzhiyun 
menu_display_statusline(struct menu * m)437*4882a593Smuzhiyun void menu_display_statusline(struct menu *m)
438*4882a593Smuzhiyun {
439*4882a593Smuzhiyun 	struct bootmenu_entry *entry;
440*4882a593Smuzhiyun 	struct bootmenu_data *menu;
441*4882a593Smuzhiyun 
442*4882a593Smuzhiyun 	if (menu_default_choice(m, (void *)&entry) < 0)
443*4882a593Smuzhiyun 		return;
444*4882a593Smuzhiyun 
445*4882a593Smuzhiyun 	menu = entry->menu;
446*4882a593Smuzhiyun 
447*4882a593Smuzhiyun 	printf(ANSI_CURSOR_POSITION, 1, 1);
448*4882a593Smuzhiyun 	puts(ANSI_CLEAR_LINE);
449*4882a593Smuzhiyun 	printf(ANSI_CURSOR_POSITION, 2, 1);
450*4882a593Smuzhiyun 	puts("  *** U-Boot Boot Menu ***");
451*4882a593Smuzhiyun 	puts(ANSI_CLEAR_LINE_TO_END);
452*4882a593Smuzhiyun 	printf(ANSI_CURSOR_POSITION, 3, 1);
453*4882a593Smuzhiyun 	puts(ANSI_CLEAR_LINE);
454*4882a593Smuzhiyun 
455*4882a593Smuzhiyun 	/* First 3 lines are bootmenu header + 2 empty lines between entries */
456*4882a593Smuzhiyun 	printf(ANSI_CURSOR_POSITION, menu->count + 5, 1);
457*4882a593Smuzhiyun 	puts(ANSI_CLEAR_LINE);
458*4882a593Smuzhiyun 	printf(ANSI_CURSOR_POSITION, menu->count + 6, 1);
459*4882a593Smuzhiyun 	puts("  Press UP/DOWN to move, ENTER to select");
460*4882a593Smuzhiyun 	puts(ANSI_CLEAR_LINE_TO_END);
461*4882a593Smuzhiyun 	printf(ANSI_CURSOR_POSITION, menu->count + 7, 1);
462*4882a593Smuzhiyun 	puts(ANSI_CLEAR_LINE);
463*4882a593Smuzhiyun }
464*4882a593Smuzhiyun 
465*4882a593Smuzhiyun #ifdef CONFIG_MENU_SHOW
menu_show(int bootdelay)466*4882a593Smuzhiyun int menu_show(int bootdelay)
467*4882a593Smuzhiyun {
468*4882a593Smuzhiyun 	bootmenu_show(bootdelay);
469*4882a593Smuzhiyun 	return -1; /* -1 - abort boot and run monitor code */
470*4882a593Smuzhiyun }
471*4882a593Smuzhiyun #endif
472*4882a593Smuzhiyun 
do_bootmenu(cmd_tbl_t * cmdtp,int flag,int argc,char * const argv[])473*4882a593Smuzhiyun int do_bootmenu(cmd_tbl_t *cmdtp, int flag, int argc, char *const argv[])
474*4882a593Smuzhiyun {
475*4882a593Smuzhiyun 	char *delay_str = NULL;
476*4882a593Smuzhiyun 	int delay = 10;
477*4882a593Smuzhiyun 
478*4882a593Smuzhiyun #if defined(CONFIG_BOOTDELAY) && (CONFIG_BOOTDELAY >= 0)
479*4882a593Smuzhiyun 	delay = CONFIG_BOOTDELAY;
480*4882a593Smuzhiyun #endif
481*4882a593Smuzhiyun 
482*4882a593Smuzhiyun 	if (argc >= 2)
483*4882a593Smuzhiyun 		delay_str = argv[1];
484*4882a593Smuzhiyun 
485*4882a593Smuzhiyun 	if (!delay_str)
486*4882a593Smuzhiyun 		delay_str = env_get("bootmenu_delay");
487*4882a593Smuzhiyun 
488*4882a593Smuzhiyun 	if (delay_str)
489*4882a593Smuzhiyun 		delay = (int)simple_strtol(delay_str, NULL, 10);
490*4882a593Smuzhiyun 
491*4882a593Smuzhiyun 	bootmenu_show(delay);
492*4882a593Smuzhiyun 	return 0;
493*4882a593Smuzhiyun }
494*4882a593Smuzhiyun 
495*4882a593Smuzhiyun U_BOOT_CMD(
496*4882a593Smuzhiyun 	bootmenu, 2, 1, do_bootmenu,
497*4882a593Smuzhiyun 	"ANSI terminal bootmenu",
498*4882a593Smuzhiyun 	"[delay]\n"
499*4882a593Smuzhiyun 	"    - show ANSI terminal bootmenu with autoboot delay"
500*4882a593Smuzhiyun );
501