scsi_id.c revision b4bbcaa9c44260e88402cb8f9a5fb8ac7f35e123
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * Copyright (C) IBM Corp. 2003
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * Copyright (C) SUSE Linux Products GmbH, 2006
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * This program is free software: you can redistribute it and/or modify
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * it under the terms of the GNU General Public License as published by
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * the Free Software Foundation, either version 2 of the License, or
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * (at your option) any later version.
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * This program is distributed in the hope that it will be useful,
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * but WITHOUT ANY WARRANTY; without even the implied warranty of
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * GNU General Public License for more details.
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * You should have received a copy of the GNU General Public License
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * along with this program. If not, see <http://www.gnu.org/licenses/>.
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering { "device", required_argument, NULL, 'd' },
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering { "config", required_argument, NULL, 'f' },
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering { "page", required_argument, NULL, 'p' },
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering { "blacklisted", no_argument, NULL, 'b' },
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering { "whitelisted", no_argument, NULL, 'g' },
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering { "replace-whitespace", no_argument, NULL, 'u' },
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering { "sg-version", required_argument, NULL, 's' },
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering { "version", no_argument, NULL, 'V' }, /* don't advertise -V */
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poetteringstatic bool all_good = false;
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poetteringstatic bool dev_specified = false;
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poetteringstatic char config_file[MAX_PATH_LEN] = "/etc/scsi_id.config";
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poetteringstatic enum page_code default_page_code = PAGE_UNSPECIFIED;
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poetteringstatic bool reformat_serial = false;
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poetteringstatic bool export = false;
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poetteringstatic void set_type(const char *from, char *to, size_t len)
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * buf points to an '=' followed by a quoted string ("foo") or a string ending
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * with a space or ','.
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * Return a pointer to the NUL terminated string, returns NULL if no
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering static const char *quote_string = "\"\n";
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * skip leading quote, terminate when quote seen
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * skip trailing quote
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * get_file_options:
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * If vendor == NULL, find a line in the config file with only "OPTIONS=";
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * if vendor and model are set find the first OPTIONS line in the config
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * file that matches. Set argc and argv to match the OPTIONS string.
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * vendor and model can end in '\n'.
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poetteringstatic int get_file_options(struct udev *udev,
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering char *vendor_in, *model_in, *options_in; /* read in from file */
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering log_error_errno(errno, "can't open %s: %m", config_file);
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * Allocate a buffer rather than put it on the stack so we can
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * keep it around to parse any options (any allocated newargv
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * points into this buffer for its strings).
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering vendor_in = model_in = options_in = NULL;
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering log_error("Config file line %d too long", lineno);
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering /* blank or all whitespace line */
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering /* comment line */
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering if (str1 && strcaseeq(str1, "OPTIONS")) {
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * Only allow: [vendor=foo[,model=bar]]options=stuff
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering if (!options_in || (!vendor_in && model_in)) {
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering log_error("Error parsing config file line %d '%s'", lineno, buffer);
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering strneq(vendor, vendor_in, strlen(vendor_in)) &&
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering (strneq(model, model_in, strlen(model_in))))) {
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * Matched vendor and optionally model.
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * Note: a short vendor_in or model_in can
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * give a partial match (that is FOO
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * matches FOOBAR).
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering if (vendor_in != NULL || model_in != NULL ||
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * Something matched. Allocate newargv, and store
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * values found in options_in.
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * argv[0] at 0 is skipped by getopt, but
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * store the buffer address there for
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * later freeing
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering /* No matches */
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poetteringstatic void help(void) {
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering printf("Usage: %s [OPTION...] DEVICE\n\n"
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering "SCSI device identification.\n\n"
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering " -h --help Print this message\n"
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering " --version Print version of the program\n\n"
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering " -d --device= Device node for SG_IO commands\n"
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering " -f --config= Location of config file\n"
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering " -p --page=0x80|0x83|pre-spc3-83 SCSI page (0x80, 0x83, pre-spc3-83)\n"
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering " -s --sg-version=3|4 Use SGv3 or SGv4\n"
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering " -b --blacklisted Treat device as blacklisted\n"
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering " -g --whitelisted Treat device as whitelisted\n"
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering " -u --replace-whitespace Replace all whitespace by underscores\n"
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering " -v --verbose Verbose logging\n"
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering " -x --export Print values as environment keys\n"
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * optind is a global extern used by getopt. Since we can call
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * set_options twice (once for command line, and once for config
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering * file) we have to reset this back to 1.
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering while ((option = getopt_long(argc, argv, "d:f:gp:uvVxh", options, NULL)) >= 0)
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering strscpy(maj_min_dev, MAX_PATH_LEN, optarg);
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering strscpy(config_file, MAX_PATH_LEN, optarg);
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering log_error("Unknown page code '%s'", optarg);
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering log_error("Unknown SG version '%s'", optarg);
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering strscpy(maj_min_dev, MAX_PATH_LEN, argv[optind]);
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poetteringstatic int per_dev_options(struct udev *udev,
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering struct scsi_id_device *dev_scsi, int *good_bad, int *page_code)
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering retval = get_file_options(udev, vendor_str, model_str, &newargc, &newargv);
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering optind = 1; /* reset this global extern */
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering while (retval == 0) {
dd0bc0f1414cc1d0fa73a29470bd14944e4942d3Lennart Poettering option = getopt_long(newargc, newargv, "bgp:", options, NULL);
if (newargv) {
return retval;
int retval;
if (retval)
return retval;
int good_dev;
int page_code;
int retval = 0;
goto out;
if (!good_dev) {
goto out;
if (export) {
goto out;
goto out;
if (reformat_serial) {
goto out;
out:
return retval;
int retval = 0;
int newargc;
log_open();
goto exit;
if (retval < 0) {
goto exit;
if (retval == 0) {
goto exit;
if (!dev_specified) {
goto exit;
exit:
if (newargv) {
log_close();
return retval;