2 * linux/fs/binfmt_script.c
4 * Copyright (C) 1996 Martin von Löwis
5 * original #!-checking implemented by tytso.
8 #include <linux/module.h>
9 #include <linux/string.h>
10 #include <linux/stat.h>
11 #include <linux/slab.h>
12 #include <linux/binfmts.h>
13 #include <linux/init.h>
14 #include <linux/file.h>
15 #include <linux/smp_lock.h>
16 #include <linux/err.h>
19 static int load_script(struct linux_binprm
*bprm
,struct pt_regs
*regs
)
21 char *cp
, *i_name
, *i_arg
;
23 char interp
[BINPRM_BUF_SIZE
];
26 if ((bprm
->buf
[0] != '#') || (bprm
->buf
[1] != '!') || (bprm
->sh_bang
))
29 * This section does the #! interpretation.
30 * Sorta complicated, but hopefully it will work. -TYT
34 allow_write_access(bprm
->file
);
38 bprm
->buf
[BINPRM_BUF_SIZE
- 1] = '\0';
39 if ((cp
= strchr(bprm
->buf
, '\n')) == NULL
)
40 cp
= bprm
->buf
+BINPRM_BUF_SIZE
-1;
42 while (cp
> bprm
->buf
) {
44 if ((*cp
== ' ') || (*cp
== '\t'))
49 for (cp
= bprm
->buf
+2; (*cp
== ' ') || (*cp
== '\t'); cp
++);
51 return -ENOEXEC
; /* No interpreter name found */
54 for ( ; *cp
&& (*cp
!= ' ') && (*cp
!= '\t'); cp
++)
56 while ((*cp
== ' ') || (*cp
== '\t'))
60 strcpy (interp
, i_name
);
62 * OK, we've parsed out the interpreter name and
63 * (optional) argument.
64 * Splice in (1) the interpreter's name for argv[0]
65 * (2) (optional) argument to interpreter
66 * (3) filename of shell script (replace argv[0])
68 * This is done in reverse order, because of how the
69 * user environment and arguments are stored.
71 remove_arg_zero(bprm
);
72 retval
= copy_strings_kernel(1, &bprm
->interp
, bprm
);
73 if (retval
< 0) return retval
;
76 retval
= copy_strings_kernel(1, &i_arg
, bprm
);
77 if (retval
< 0) return retval
;
80 retval
= copy_strings_kernel(1, &i_name
, bprm
);
81 if (retval
) return retval
;
83 bprm
->interp
= interp
;
86 * OK, now restart the process with the interpreter's dentry.
88 file
= open_exec(interp
);
93 retval
= prepare_binprm(bprm
);
96 return search_binary_handler(bprm
,regs
);
99 static struct linux_binfmt script_format
= {
100 .module
= THIS_MODULE
,
101 .load_binary
= load_script
,
104 static int __init
init_script_binfmt(void)
106 return register_binfmt(&script_format
);
109 static void __exit
exit_script_binfmt(void)
111 unregister_binfmt(&script_format
);
114 core_initcall(init_script_binfmt
);
115 module_exit(exit_script_binfmt
);
116 MODULE_LICENSE("GPL");