1 /*
2  * Copyright (C) 2013 Michael Brown <mbrown@fensystems.co.uk>.
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License as
6  * published by the Free Software Foundation; either version 2 of the
7  * License, or any later version.
8  *
9  * This program is distributed in the hope that it will be useful, but
10  * WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
17  * 02110-1301, USA.
18  *
19  * You can also choose to distribute this program under the terms of
20  * the Unmodified Binary Distribution Licence (as given in the file
21  * COPYING.UBDL), provided that you have satisfied its requirements.
22  */
23 
24 #include <stdio.h>
25 #include <getopt.h>
26 #include <ipxe/pci.h>
27 #include <ipxe/command.h>
28 #include <ipxe/parseopt.h>
29 
30 FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
31 
32 /** @file
33  *
34  * PCI commands
35  *
36  */
37 
38 /** "pciscan" options */
39 struct pciscan_options {};
40 
41 /** "pciscan" option list */
42 static struct option_descriptor pciscan_opts[] = {};
43 
44 /** "pciscan" command descriptor */
45 static struct command_descriptor pciscan_cmd =
46 	COMMAND_DESC ( struct pciscan_options, pciscan_opts, 1, 1,
47 		       "<setting>" );
48 
49 /**
50  * "pciscan" command
51  *
52  * @v argc		Argument count
53  * @v argv		Argument list
54  * @ret rc		Return status code
55  */
pciscan_exec(int argc,char ** argv)56 static int pciscan_exec ( int argc, char **argv ) {
57 	struct pciscan_options opts;
58 	struct named_setting setting;
59 	struct pci_device pci;
60 	unsigned long prev;
61 	int next;
62 	int len;
63 	int rc;
64 
65 	/* Parse options */
66 	if ( ( rc = parse_options ( argc, argv, &pciscan_cmd, &opts ) ) != 0 )
67 		goto err_parse_options;
68 
69 	/* Parse setting name */
70 	if ( ( rc = parse_autovivified_setting ( argv[optind],
71 						 &setting ) ) != 0 )
72 		goto err_parse_setting;
73 
74 	/* Determine starting bus:dev.fn address */
75 	if ( ( len = fetchn_setting ( setting.settings, &setting.setting,
76 				      NULL, &setting.setting, &prev ) ) < 0 ) {
77 		/* Setting not yet defined: start searching from 00:00.0 */
78 		prev = 0;
79 	} else {
80 		/* Setting is defined: start searching from next location */
81 		prev++;
82 	}
83 
84 	/* Find next existent PCI device */
85 	if ( ( next = pci_find_next ( &pci, prev ) ) < 0 ) {
86 		rc = next;
87 		goto err_find_next;
88 	}
89 
90 	/* Apply default type if necessary.  Use ":uint16" rather than
91 	 * ":busdevfn" to allow for easy inclusion within a
92 	 * "${pci/${location}.x.y}" constructed setting.
93 	 */
94 	if ( ! setting.setting.type )
95 		setting.setting.type = &setting_type_uint16;
96 
97 	/* Store setting */
98 	if ( ( rc = storen_setting ( setting.settings, &setting.setting,
99 				     next ) ) != 0 ) {
100 		printf ( "Could not store \"%s\": %s\n",
101 			 setting.setting.name, strerror ( rc ) );
102 		goto err_store;
103 	}
104 
105  err_store:
106  err_find_next:
107  err_parse_setting:
108  err_parse_options:
109 	return rc;
110 }
111 
112 /** PCI commands */
113 struct command pci_commands[] __command = {
114 	{
115 		.name = "pciscan",
116 		.exec = pciscan_exec,
117 	},
118 };
119