mirror of https://github.com/ipxe/ipxe.git
487 lines
9.5 KiB
C
487 lines
9.5 KiB
C
/*
|
|
* Copyright (C) 2007 Michael Brown <mbrown@fensystems.co.uk>.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
* License, or any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*/
|
|
|
|
#include <stdint.h>
|
|
#include <stdlib.h>
|
|
#include <libgen.h>
|
|
#include <getopt.h>
|
|
#include <vsprintf.h>
|
|
#include <gpxe/image.h>
|
|
#include <gpxe/command.h>
|
|
#include <usr/imgmgmt.h>
|
|
|
|
/** @file
|
|
*
|
|
* Image management commands
|
|
*
|
|
*/
|
|
|
|
/**
|
|
* Print image description
|
|
*
|
|
*/
|
|
|
|
/**
|
|
* Fill in image command line
|
|
*
|
|
* @v image Image
|
|
* @v nargs Argument count
|
|
* @v args Argument list
|
|
*/
|
|
void imgfill_cmdline ( struct image *image, unsigned int nargs, char **args ) {
|
|
size_t used = 0;
|
|
|
|
image->cmdline[0] = '\0';
|
|
while ( ( used < sizeof ( image->cmdline ) ) && nargs-- ) {
|
|
used += snprintf ( &image->cmdline[used],
|
|
( sizeof ( image->cmdline ) - used ),
|
|
"%s%s", ( used ? " " : "" ), *(args++) );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* "imgfetch"/"module"/"kernel" command syntax message
|
|
*
|
|
* @v argv Argument list
|
|
*/
|
|
static void imgfetch_core_syntax ( char **argv, int load ) {
|
|
printf ( "Usage:\n"
|
|
" %s [-n|--name <name>] filename [arguments...]\n"
|
|
"\n"
|
|
"%s executable/loadable image\n",
|
|
argv[0], ( load ? "Fetch and load" : "Fetch" ) );
|
|
}
|
|
|
|
/**
|
|
* The "imgfetch"/"module"/"kernel" command body
|
|
*
|
|
* @v argc Argument count
|
|
* @v argv Argument list
|
|
* @v load Load image after fetching
|
|
* @ret rc Exit code
|
|
*/
|
|
static int imgfetch_core_exec ( int argc, char **argv, int load ) {
|
|
static struct option longopts[] = {
|
|
{ "help", 0, NULL, 'h' },
|
|
{ "name", required_argument, NULL, 'n' },
|
|
{ NULL, 0, NULL, 0 },
|
|
};
|
|
struct image *image;
|
|
const char *name = NULL;
|
|
char *filename;
|
|
int c;
|
|
int rc;
|
|
|
|
/* Parse options */
|
|
while ( ( c = getopt_long ( argc, argv, "hn:",
|
|
longopts, NULL ) ) >= 0 ) {
|
|
switch ( c ) {
|
|
case 'n':
|
|
/* Set image name */
|
|
name = optarg;
|
|
break;
|
|
case 'h':
|
|
/* Display help text */
|
|
default:
|
|
/* Unrecognised/invalid option */
|
|
imgfetch_core_syntax ( argv, load );
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
/* Need at least a filename remaining after the options */
|
|
if ( optind == argc ) {
|
|
imgfetch_core_syntax ( argv, load );
|
|
return 1;
|
|
}
|
|
filename = argv[optind++];
|
|
if ( ! name )
|
|
name = basename ( filename );
|
|
|
|
/* Fetch the image */
|
|
if ( ( rc = imgfetch ( filename, name, &image ) ) != 0 ) {
|
|
printf ( "Could not fetch %s: %s\n", name, strerror ( rc ) );
|
|
return 1;
|
|
}
|
|
|
|
/* Fill in command line */
|
|
imgfill_cmdline ( image, ( argc - optind ), &argv[optind] );
|
|
|
|
/* Load image if required */
|
|
if ( load ) {
|
|
if ( ( rc = imgload ( image ) ) != 0 ) {
|
|
printf ( "Could not load %s: %s\n", name,
|
|
strerror ( rc ) );
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* The "imgfetch"/"module" command
|
|
*
|
|
* @v argc Argument count
|
|
* @v argv Argument list
|
|
* @ret rc Exit code
|
|
*/
|
|
static int imgfetch_exec ( int argc, char **argv ) {
|
|
return imgfetch_core_exec ( argc, argv, 0 );
|
|
}
|
|
|
|
/**
|
|
* The "kernel" command
|
|
*
|
|
* @v argc Argument count
|
|
* @v argv Argument list
|
|
* @ret rc Exit code
|
|
*/
|
|
static int kernel_exec ( int argc, char **argv ) {
|
|
return imgfetch_core_exec ( argc, argv, 1 );
|
|
}
|
|
|
|
/**
|
|
* "imgload" command syntax message
|
|
*
|
|
* @v argv Argument list
|
|
*/
|
|
static void imgload_syntax ( char **argv ) {
|
|
printf ( "Usage:\n"
|
|
" %s <image name>\n"
|
|
"\n"
|
|
"Load executable/loadable image\n",
|
|
argv[0] );
|
|
}
|
|
|
|
/**
|
|
* The "imgload" command
|
|
*
|
|
* @v argc Argument count
|
|
* @v argv Argument list
|
|
* @ret rc Exit code
|
|
*/
|
|
static int imgload_exec ( int argc, char **argv ) {
|
|
static struct option longopts[] = {
|
|
{ "help", 0, NULL, 'h' },
|
|
{ NULL, 0, NULL, 0 },
|
|
};
|
|
struct image *image;
|
|
const char *name;
|
|
int c;
|
|
int rc;
|
|
|
|
/* Parse options */
|
|
while ( ( c = getopt_long ( argc, argv, "h", longopts, NULL ) ) >= 0 ){
|
|
switch ( c ) {
|
|
case 'h':
|
|
/* Display help text */
|
|
default:
|
|
/* Unrecognised/invalid option */
|
|
imgload_syntax ( argv );
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
/* Need exactly one image name remaining after the options */
|
|
if ( optind != ( argc - 1 ) ) {
|
|
imgload_syntax ( argv );
|
|
return 1;
|
|
}
|
|
name = argv[optind];
|
|
|
|
/* Load all specified images */
|
|
image = find_image ( name );
|
|
if ( ! image ) {
|
|
printf ( "No such image: %s\n", name );
|
|
return 1;
|
|
}
|
|
if ( ( rc = imgload ( image ) ) != 0 ) {
|
|
printf ( "Could not load %s: %s\n", name, strerror ( rc ) );
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* "imgargs" command syntax message
|
|
*
|
|
* @v argv Argument list
|
|
*/
|
|
static void imgargs_syntax ( char **argv ) {
|
|
printf ( "Usage:\n"
|
|
" %s <image name> [<arguments>...]\n"
|
|
"\n"
|
|
"Set arguments for executable/loadable image\n",
|
|
argv[0] );
|
|
}
|
|
|
|
/**
|
|
* The "imgargs" command body
|
|
*
|
|
* @v argc Argument count
|
|
* @v argv Argument list
|
|
* @ret rc Exit code
|
|
*/
|
|
static int imgargs_exec ( int argc, char **argv ) {
|
|
static struct option longopts[] = {
|
|
{ "help", 0, NULL, 'h' },
|
|
{ NULL, 0, NULL, 0 },
|
|
};
|
|
struct image *image;
|
|
const char *name;
|
|
int c;
|
|
|
|
/* Parse options */
|
|
while ( ( c = getopt_long ( argc, argv, "h", longopts, NULL ) ) >= 0 ){
|
|
switch ( c ) {
|
|
case 'h':
|
|
/* Display help text */
|
|
default:
|
|
/* Unrecognised/invalid option */
|
|
imgargs_syntax ( argv );
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
/* Need at least an image name remaining after the options */
|
|
if ( optind == argc ) {
|
|
imgargs_syntax ( argv );
|
|
return 1;
|
|
}
|
|
name = argv[optind++];
|
|
|
|
/* Fill in command line */
|
|
image = find_image ( name );
|
|
if ( ! image ) {
|
|
printf ( "No such image: %s\n", name );
|
|
return 1;
|
|
}
|
|
imgfill_cmdline ( image, ( argc - optind ), &argv[optind] );
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* "imgexec" command syntax message
|
|
*
|
|
* @v argv Argument list
|
|
*/
|
|
static void imgexec_syntax ( char **argv ) {
|
|
printf ( "Usage:\n"
|
|
" %s <image name>\n"
|
|
"\n"
|
|
"Execute executable/loadable image\n",
|
|
argv[0] );
|
|
}
|
|
|
|
/**
|
|
* The "imgexec" command
|
|
*
|
|
* @v argc Argument count
|
|
* @v argv Argument list
|
|
* @ret rc Exit code
|
|
*/
|
|
static int imgexec_exec ( int argc, char **argv ) {
|
|
static struct option longopts[] = {
|
|
{ "help", 0, NULL, 'h' },
|
|
{ NULL, 0, NULL, 0 },
|
|
};
|
|
struct image *image;
|
|
const char *name;
|
|
int c;
|
|
int rc;
|
|
|
|
/* Parse options */
|
|
while ( ( c = getopt_long ( argc, argv, "h", longopts, NULL ) ) >= 0 ){
|
|
switch ( c ) {
|
|
case 'h':
|
|
/* Display help text */
|
|
default:
|
|
/* Unrecognised/invalid option */
|
|
imgexec_syntax ( argv );
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
/* Need exactly one image name */
|
|
if ( optind != ( argc - 1 ) ) {
|
|
imgexec_syntax ( argv );
|
|
return 1;
|
|
}
|
|
name = argv[optind];
|
|
|
|
/* Execute specified image */
|
|
image = find_image ( name );
|
|
if ( ! image ) {
|
|
printf ( "No such image: %s\n", name );
|
|
return 1;
|
|
}
|
|
if ( ( rc = imgexec ( image ) ) != 0 ) {
|
|
printf ( "Could not execute %s: %s\n", name, strerror ( rc ) );
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* "imgstat" command syntax message
|
|
*
|
|
* @v argv Argument list
|
|
*/
|
|
static void imgstat_syntax ( char **argv ) {
|
|
printf ( "Usage:\n"
|
|
" %s\n"
|
|
"\n"
|
|
"List executable/loadable images\n",
|
|
argv[0] );
|
|
}
|
|
|
|
/**
|
|
* The "imgstat" command
|
|
*
|
|
* @v argc Argument count
|
|
* @v argv Argument list
|
|
* @ret rc Exit code
|
|
*/
|
|
static int imgstat_exec ( int argc, char **argv ) {
|
|
static struct option longopts[] = {
|
|
{ "help", 0, NULL, 'h' },
|
|
{ NULL, 0, NULL, 0 },
|
|
};
|
|
struct image *image;
|
|
int c;
|
|
|
|
/* Parse options */
|
|
while ( ( c = getopt_long ( argc, argv, "h", longopts, NULL ) ) >= 0 ){
|
|
switch ( c ) {
|
|
case 'h':
|
|
/* Display help text */
|
|
default:
|
|
/* Unrecognised/invalid option */
|
|
imgstat_syntax ( argv );
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
/* No arguments */
|
|
if ( optind != argc ) {
|
|
imgstat_syntax ( argv );
|
|
return 1;
|
|
}
|
|
|
|
/* Show status of all images */
|
|
for_each_image ( image ) {
|
|
imgstat ( image );
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* "imgstat" command syntax message
|
|
*
|
|
* @v argv Argument list
|
|
*/
|
|
static void imgfree_syntax ( char **argv ) {
|
|
printf ( "Usage:\n"
|
|
" %s\n"
|
|
"\n"
|
|
"Free all executable/loadable images\n",
|
|
argv[0] );
|
|
}
|
|
|
|
/**
|
|
* The "imgfree" command
|
|
*
|
|
* @v argc Argument count
|
|
* @v argv Argument list
|
|
* @ret rc Exit code
|
|
*/
|
|
static int imgfree_exec ( int argc, char **argv ) {
|
|
static struct option longopts[] = {
|
|
{ "help", 0, NULL, 'h' },
|
|
{ NULL, 0, NULL, 0 },
|
|
};
|
|
struct image *image;
|
|
struct image *tmp;
|
|
int c;
|
|
|
|
/* Parse options */
|
|
while ( ( c = getopt_long ( argc, argv, "h", longopts, NULL ) ) >= 0 ){
|
|
switch ( c ) {
|
|
case 'h':
|
|
/* Display help text */
|
|
default:
|
|
/* Unrecognised/invalid option */
|
|
imgfree_syntax ( argv );
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
/* No arguments */
|
|
if ( optind != argc ) {
|
|
imgfree_syntax ( argv );
|
|
return 1;
|
|
}
|
|
|
|
/* Free all images */
|
|
list_for_each_entry_safe ( image, tmp, &images, list ) {
|
|
imgfree ( image );
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/** Image management commands */
|
|
struct command image_commands[] __command = {
|
|
{
|
|
.name = "imgfetch",
|
|
.exec = imgfetch_exec,
|
|
},
|
|
{
|
|
.name = "module",
|
|
.exec = imgfetch_exec, /* synonym for "imgfetch" */
|
|
},
|
|
{
|
|
.name = "kernel",
|
|
.exec = kernel_exec,
|
|
},
|
|
{
|
|
.name = "imgload",
|
|
.exec = imgload_exec,
|
|
},
|
|
{
|
|
.name = "imgargs",
|
|
.exec = imgargs_exec,
|
|
},
|
|
{
|
|
.name = "imgexec",
|
|
.exec = imgexec_exec,
|
|
},
|
|
{
|
|
.name = "imgstat",
|
|
.exec = imgstat_exec,
|
|
},
|
|
{
|
|
.name = "imgfree",
|
|
.exec = imgfree_exec,
|
|
},
|
|
};
|