blob: c4229719e2617828256b83fa9a5fa1014443e2bb [file] [log] [blame]
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -02001/*
2 * kmod-insmod - insert modules into linux kernel using libkmod.
3 *
Lucas De Marchie6b0e492013-01-16 11:27:21 -02004 * Copyright (C) 2011-2013 ProFUSION embedded systems
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -02005 *
Lucas De Marchicb451f32011-12-12 18:24:35 -02006 * This program is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 2 of the License, or
9 * (at your option) any later version.
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020010 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Lucas De Marchicb451f32011-12-12 18:24:35 -020013 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020015 *
Lucas De Marchicb451f32011-12-12 18:24:35 -020016 * You should have received a copy of the GNU General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020018 */
Lucas De Marchicb451f32011-12-12 18:24:35 -020019
Lucas De Marchic2e42862014-10-03 01:41:42 -030020#include <errno.h>
21#include <getopt.h>
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020022#include <stdio.h>
23#include <stdlib.h>
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020024#include <string.h>
Lucas De Marchic2e42862014-10-03 01:41:42 -030025
Lucas De Marchi5b0436a2015-01-14 14:22:23 -020026#include <shared/util.h>
27
Lucas De Marchif3578662015-01-02 12:38:27 -020028#include <libkmod/libkmod.h>
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020029
Lucas De Marchi4a2e20d2012-11-06 16:54:17 -020030#include "kmod.h"
31
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020032static const char cmdopts_s[] = "psfVh";
33static const struct option cmdopts[] = {
34 {"version", no_argument, 0, 'V'},
35 {"help", no_argument, 0, 'h'},
36 {NULL, 0, 0, 0}
37};
38
Lucas De Marchi4a2e20d2012-11-06 16:54:17 -020039static void help(void)
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020040{
Lucas De Marchi34e06bf2012-11-06 17:32:41 -020041 printf("Usage:\n"
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020042 "\t%s [options] filename [args]\n"
43 "Options:\n"
44 "\t-V, --version show version\n"
45 "\t-h, --help show this help\n",
Lucas De Marchi7c04aee2012-11-06 19:20:09 -020046 program_invocation_short_name);
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020047}
48
49static const char *mod_strerror(int err)
50{
51 switch (err) {
52 case ENOEXEC:
53 return "Invalid module format";
54 case ENOENT:
55 return "Unknown symbol in module";
56 case ESRCH:
57 return "Module has wrong symbol version";
58 case EINVAL:
59 return "Invalid parameters";
60 default:
61 return strerror(err);
62 }
63}
64
Lucas De Marchiad602692011-12-22 03:45:07 -020065static int do_insmod(int argc, char *argv[])
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020066{
67 struct kmod_ctx *ctx;
68 struct kmod_module *mod;
69 const char *filename;
70 char *opts = NULL;
71 size_t optslen = 0;
72 int i, err;
Lucas De Marchi2411c072011-12-12 15:41:02 -020073 const char *null_config = NULL;
Philippe De Swert2b2a5032015-10-07 18:36:58 +030074 unsigned int flags = 0;
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020075
76 for (;;) {
77 int c, idx = 0;
78 c = getopt_long(argc, argv, cmdopts_s, cmdopts, &idx);
79 if (c == -1)
80 break;
81 switch (c) {
82 case 'p':
83 case 's':
Marc-Antoine Perennoud64b2862015-11-20 22:23:14 +010084 /* ignored, for compatibility only */
85 break;
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020086 case 'f':
Philippe De Swert2b2a5032015-10-07 18:36:58 +030087 flags |= KMOD_PROBE_FORCE_MODVERSION;
88 flags |= KMOD_PROBE_FORCE_VERMAGIC;
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020089 break;
90 case 'h':
Lucas De Marchi4a2e20d2012-11-06 16:54:17 -020091 help();
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020092 return EXIT_SUCCESS;
93 case 'V':
94 puts(PACKAGE " version " VERSION);
Lucas De Marchi655de272015-06-07 02:46:22 -030095 puts(KMOD_FEATURES);
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -020096 return EXIT_SUCCESS;
97 case '?':
98 return EXIT_FAILURE;
99 default:
Lucas De Marchie2f94782012-11-05 15:08:54 -0200100 ERR("unexpected getopt_long() value '%c'.\n",
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -0200101 c);
102 return EXIT_FAILURE;
103 }
104 }
105
106 if (optind >= argc) {
Lucas De Marchie2f94782012-11-05 15:08:54 -0200107 ERR("missing filename.\n");
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -0200108 return EXIT_FAILURE;
109 }
110
111 filename = argv[optind];
Lucas De Marchi5b0436a2015-01-14 14:22:23 -0200112 if (streq(filename, "-")) {
Lucas De Marchie2f94782012-11-05 15:08:54 -0200113 ERR("this tool does not support loading from stdin!\n");
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -0200114 return EXIT_FAILURE;
115 }
116
117 for (i = optind + 1; i < argc; i++) {
118 size_t len = strlen(argv[i]);
119 void *tmp = realloc(opts, optslen + len + 2);
120 if (tmp == NULL) {
Lucas De Marchie2f94782012-11-05 15:08:54 -0200121 ERR("out of memory\n");
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -0200122 free(opts);
123 return EXIT_FAILURE;
124 }
125 opts = tmp;
126 if (optslen > 0) {
127 opts[optslen] = ' ';
128 optslen++;
129 }
130 memcpy(opts + optslen, argv[i], len);
131 optslen += len;
132 opts[optslen] = '\0';
133 }
134
Lucas De Marchi2411c072011-12-12 15:41:02 -0200135 ctx = kmod_new(NULL, &null_config);
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -0200136 if (!ctx) {
Lucas De Marchie2f94782012-11-05 15:08:54 -0200137 ERR("kmod_new() failed!\n");
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -0200138 free(opts);
139 return EXIT_FAILURE;
140 }
141
142 err = kmod_module_new_from_path(ctx, filename, &mod);
143 if (err < 0) {
Lucas De Marchie2f94782012-11-05 15:08:54 -0200144 ERR("could not load module %s: %s\n", filename,
145 strerror(-err));
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -0200146 goto end;
147 }
148
Philippe De Swert2b2a5032015-10-07 18:36:58 +0300149 err = kmod_module_insert_module(mod, flags, opts);
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -0200150 if (err < 0) {
Lucas De Marchie2f94782012-11-05 15:08:54 -0200151 ERR("could not insert module %s: %s\n", filename,
152 mod_strerror(-err));
Gustavo Sverzut Barbieri72c51a92011-12-10 22:19:41 -0200153 }
154 kmod_module_unref(mod);
155
156end:
157 kmod_unref(ctx);
158 free(opts);
159 return err >= 0 ? EXIT_SUCCESS : EXIT_FAILURE;
160}
Lucas De Marchiad602692011-12-22 03:45:07 -0200161
Lucas De Marchiad602692011-12-22 03:45:07 -0200162const struct kmod_cmd kmod_cmd_compat_insmod = {
163 .name = "insmod",
164 .cmd = do_insmod,
165 .help = "compat insmod command",
166};