Check for errors when registering and unregistering firmware modules. This

prevents the unfortunate situation whereby one might kldload a firmware
module by hand and later kldunload it while a driver still has an open
reference to it.

MFC after:	1 week
Glanced at by:	iedowse
This commit is contained in:
jhb 2006-06-16 21:06:03 +00:00
parent 2e1fa1a727
commit 5094bdb85f

View File

@ -125,7 +125,8 @@ for (file_i = 0; file_i < num_files; file_i++) {
printc("\nstatic int\n"\ printc("\nstatic int\n"\
opt_m "_fw_modevent(module_t mod, int type, void *unused)\ opt_m "_fw_modevent(module_t mod, int type, void *unused)\
{\ {\
struct firmware *fp;\ struct firmware *fp, *parent;\
int error;\
switch (type) {\ switch (type) {\
case MOD_LOAD:"); case MOD_LOAD:");
@ -136,11 +137,7 @@ for (file_i = 0; file_i < num_files; file_i++) {
# '-', '.' and '/' are converted to '_' by ld/objcopy # '-', '.' and '/' are converted to '_' by ld/objcopy
gsub(/-|\.|\//, "_", symb); gsub(/-|\.|\//, "_", symb);
if (file_i == 0) reg = "\t\tfp = ";
reg = "\t\tfp = ";
else
reg = "\t\t(void)";
reg = reg "firmware_register(\"" short "\", _binary_" symb "_start , "; reg = reg "firmware_register(\"" short "\", _binary_" symb "_start , ";
reg = reg "(size_t)(_binary_" symb "_end - _binary_" symb "_start), "; reg = reg "(size_t)(_binary_" symb "_end - _binary_" symb "_start), ";
reg = reg version ", "; reg = reg version ", ";
@ -148,21 +145,37 @@ for (file_i = 0; file_i < num_files; file_i++) {
if (file_i == 0) if (file_i == 0)
reg = reg "NULL);"; reg = reg "NULL);";
else else
reg = reg "fp);"; reg = reg "parent);";
printc(reg); printc(reg);
printc("\t\tif (fp == NULL)");
printc("\t\t\tgoto fail_" file_i ";");
if (file_i == 0)
printc("\t\tparent = fp;");
} }
printc("\t\treturn (0);\ printc("\t\treturn (0);");
case MOD_UNLOAD:");
for (file_i = num_files - 1; file_i > 0; file_i--) {
printc("\tfail_" file_i ":")
printc("\t\t(void)firmware_unregister(\"" shortnames[file_i - 1] "\");");
}
printc("\tfail_0:");
printc("\t\treturn (ENXIO);");
printc("\tcase MOD_UNLOAD:");
for (file_i = 1; file_i < num_files; file_i++) { for (file_i = 1; file_i < num_files; file_i++) {
printc("\t\tfirmware_unregister(\"" shortnames[file_i] "\");"); printc("\t\terror = firmware_unregister(\"" shortnames[file_i] "\");");
printc("\t\tif (error)");
printc("\t\t\treturn (error);");
} }
printc("\t\tfirmware_unregister(\"" shortnames[0] "\");"); printc("\t\terror = firmware_unregister(\"" shortnames[0] "\");");
printc("\t\treturn (0);\ printc("\t\treturn (error);\
}\ }\
return (EINVAL);\ return (EINVAL);\
}\ }\