binfmt_som.c 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. /*
  2. * linux/fs/binfmt_som.c
  3. *
  4. * These are the functions used to load SOM format executables as used
  5. * by HP-UX.
  6. *
  7. * Copyright 1999 Matthew Wilcox <willy@bofh.ai>
  8. * based on binfmt_elf which is
  9. * Copyright 1993, 1994: Eric Youngdale (ericy@cais.com).
  10. */
  11. #include <linux/module.h>
  12. #include <linux/fs.h>
  13. #include <linux/stat.h>
  14. #include <linux/sched.h>
  15. #include <linux/mm.h>
  16. #include <linux/mman.h>
  17. #include <linux/errno.h>
  18. #include <linux/signal.h>
  19. #include <linux/binfmts.h>
  20. #include <linux/som.h>
  21. #include <linux/string.h>
  22. #include <linux/file.h>
  23. #include <linux/fcntl.h>
  24. #include <linux/ptrace.h>
  25. #include <linux/slab.h>
  26. #include <linux/shm.h>
  27. #include <linux/personality.h>
  28. #include <linux/init.h>
  29. #include <asm/uaccess.h>
  30. #include <asm/pgtable.h>
  31. #include <linux/elf.h>
  32. static int load_som_binary(struct linux_binprm * bprm, struct pt_regs * regs);
  33. static int load_som_library(struct file *);
  34. /*
  35. * If we don't support core dumping, then supply a NULL so we
  36. * don't even try.
  37. */
  38. #if 0
  39. static int som_core_dump(struct coredump_params *cprm);
  40. #else
  41. #define som_core_dump NULL
  42. #endif
  43. #define SOM_PAGESTART(_v) ((_v) & ~(unsigned long)(SOM_PAGESIZE-1))
  44. #define SOM_PAGEOFFSET(_v) ((_v) & (SOM_PAGESIZE-1))
  45. #define SOM_PAGEALIGN(_v) (((_v) + SOM_PAGESIZE - 1) & ~(SOM_PAGESIZE - 1))
  46. static struct linux_binfmt som_format = {
  47. .module = THIS_MODULE,
  48. .load_binary = load_som_binary,
  49. .load_shlib = load_som_library,
  50. .core_dump = som_core_dump,
  51. .min_coredump = SOM_PAGESIZE
  52. };
  53. /*
  54. * create_som_tables() parses the env- and arg-strings in new user
  55. * memory and creates the pointer tables from them, and puts their
  56. * addresses on the "stack", returning the new stack pointer value.
  57. */
  58. static void create_som_tables(struct linux_binprm *bprm)
  59. {
  60. char **argv, **envp;
  61. int argc = bprm->argc;
  62. int envc = bprm->envc;
  63. unsigned long p;
  64. unsigned long *sp;
  65. /* Word-align the stack pointer */
  66. sp = (unsigned long *)((bprm->p + 3) & ~3);
  67. envp = (char **) sp;
  68. sp += envc + 1;
  69. argv = (char **) sp;
  70. sp += argc + 1;
  71. __put_user((unsigned long) envp,++sp);
  72. __put_user((unsigned long) argv,++sp);
  73. __put_user(argc, ++sp);
  74. bprm->p = (unsigned long) sp;
  75. p = current->mm->arg_start;
  76. while (argc-- > 0) {
  77. __put_user((char *)p,argv++);
  78. p += strlen_user((char *)p);
  79. }
  80. __put_user(NULL, argv);
  81. current->mm->arg_end = current->mm->env_start = p;
  82. while (envc-- > 0) {
  83. __put_user((char *)p,envp++);
  84. p += strlen_user((char *)p);
  85. }
  86. __put_user(NULL, envp);
  87. current->mm->env_end = p;
  88. }
  89. static int check_som_header(struct som_hdr *som_ex)
  90. {
  91. int *buf = (int *)som_ex;
  92. int i, ck;
  93. if (som_ex->system_id != SOM_SID_PARISC_1_0 &&
  94. som_ex->system_id != SOM_SID_PARISC_1_1 &&
  95. som_ex->system_id != SOM_SID_PARISC_2_0)
  96. return -ENOEXEC;
  97. if (som_ex->a_magic != SOM_EXEC_NONSHARE &&
  98. som_ex->a_magic != SOM_EXEC_SHARE &&
  99. som_ex->a_magic != SOM_EXEC_DEMAND)
  100. return -ENOEXEC;
  101. if (som_ex->version_id != SOM_ID_OLD &&
  102. som_ex->version_id != SOM_ID_NEW)
  103. return -ENOEXEC;
  104. ck = 0;
  105. for (i=0; i<32; i++)
  106. ck ^= buf[i];
  107. if (ck != 0)
  108. return -ENOEXEC;
  109. return 0;
  110. }
  111. static int map_som_binary(struct file *file,
  112. const struct som_exec_auxhdr *hpuxhdr)
  113. {
  114. unsigned long code_start, code_size, data_start, data_size;
  115. unsigned long bss_start, som_brk;
  116. int retval;
  117. int prot = PROT_READ | PROT_EXEC;
  118. int flags = MAP_FIXED|MAP_PRIVATE|MAP_DENYWRITE|MAP_EXECUTABLE;
  119. mm_segment_t old_fs = get_fs();
  120. set_fs(get_ds());
  121. code_start = SOM_PAGESTART(hpuxhdr->exec_tmem);
  122. code_size = SOM_PAGEALIGN(hpuxhdr->exec_tsize);
  123. current->mm->start_code = code_start;
  124. current->mm->end_code = code_start + code_size;
  125. retval = vm_mmap(file, code_start, code_size, prot,
  126. flags, SOM_PAGESTART(hpuxhdr->exec_tfile));
  127. if (retval < 0 && retval > -1024)
  128. goto out;
  129. data_start = SOM_PAGESTART(hpuxhdr->exec_dmem);
  130. data_size = SOM_PAGEALIGN(hpuxhdr->exec_dsize);
  131. current->mm->start_data = data_start;
  132. current->mm->end_data = bss_start = data_start + data_size;
  133. retval = vm_mmap(file, data_start, data_size,
  134. prot | PROT_WRITE, flags,
  135. SOM_PAGESTART(hpuxhdr->exec_dfile));
  136. if (retval < 0 && retval > -1024)
  137. goto out;
  138. som_brk = bss_start + SOM_PAGEALIGN(hpuxhdr->exec_bsize);
  139. current->mm->start_brk = current->mm->brk = som_brk;
  140. retval = vm_mmap(NULL, bss_start, som_brk - bss_start,
  141. prot | PROT_WRITE, MAP_FIXED | MAP_PRIVATE, 0);
  142. if (retval > 0 || retval < -1024)
  143. retval = 0;
  144. out:
  145. set_fs(old_fs);
  146. return retval;
  147. }
  148. /*
  149. * These are the functions used to load SOM executables and shared
  150. * libraries. There is no binary dependent code anywhere else.
  151. */
  152. static int
  153. load_som_binary(struct linux_binprm * bprm, struct pt_regs * regs)
  154. {
  155. int retval;
  156. unsigned int size;
  157. unsigned long som_entry;
  158. struct som_hdr *som_ex;
  159. struct som_exec_auxhdr *hpuxhdr;
  160. /* Get the exec-header */
  161. som_ex = (struct som_hdr *) bprm->buf;
  162. retval = check_som_header(som_ex);
  163. if (retval != 0)
  164. goto out;
  165. /* Now read in the auxiliary header information */
  166. retval = -ENOMEM;
  167. size = som_ex->aux_header_size;
  168. if (size > SOM_PAGESIZE)
  169. goto out;
  170. hpuxhdr = kmalloc(size, GFP_KERNEL);
  171. if (!hpuxhdr)
  172. goto out;
  173. retval = kernel_read(bprm->file, som_ex->aux_header_location,
  174. (char *) hpuxhdr, size);
  175. if (retval != size) {
  176. if (retval >= 0)
  177. retval = -EIO;
  178. goto out_free;
  179. }
  180. /* Flush all traces of the currently running executable */
  181. retval = flush_old_exec(bprm);
  182. if (retval)
  183. goto out_free;
  184. /* OK, This is the point of no return */
  185. current->personality = PER_HPUX;
  186. setup_new_exec(bprm);
  187. /* Set the task size for HP-UX processes such that
  188. * the gateway page is outside the address space.
  189. * This can be fixed later, but for now, this is much
  190. * easier.
  191. */
  192. current->thread.task_size = 0xc0000000;
  193. /* Set map base to allow enough room for hp-ux heap growth */
  194. current->thread.map_base = 0x80000000;
  195. retval = map_som_binary(bprm->file, hpuxhdr);
  196. if (retval < 0)
  197. goto out_free;
  198. som_entry = hpuxhdr->exec_entry;
  199. kfree(hpuxhdr);
  200. set_binfmt(&som_format);
  201. install_exec_creds(bprm);
  202. setup_arg_pages(bprm, STACK_TOP, EXSTACK_DEFAULT);
  203. create_som_tables(bprm);
  204. current->mm->start_stack = bprm->p;
  205. #if 0
  206. printk("(start_brk) %08lx\n" , (unsigned long) current->mm->start_brk);
  207. printk("(end_code) %08lx\n" , (unsigned long) current->mm->end_code);
  208. printk("(start_code) %08lx\n" , (unsigned long) current->mm->start_code);
  209. printk("(end_data) %08lx\n" , (unsigned long) current->mm->end_data);
  210. printk("(start_stack) %08lx\n" , (unsigned long) current->mm->start_stack);
  211. printk("(brk) %08lx\n" , (unsigned long) current->mm->brk);
  212. #endif
  213. map_hpux_gateway_page(current,current->mm);
  214. start_thread_som(regs, som_entry, bprm->p);
  215. return 0;
  216. /* error cleanup */
  217. out_free:
  218. kfree(hpuxhdr);
  219. out:
  220. return retval;
  221. }
  222. static int load_som_library(struct file *f)
  223. {
  224. /* No lib support in SOM yet. gizza chance.. */
  225. return -ENOEXEC;
  226. }
  227. /* Install the SOM loader.
  228. * N.B. We *rely* on the table being the right size with the
  229. * right number of free slots...
  230. */
  231. static int __init init_som_binfmt(void)
  232. {
  233. register_binfmt(&som_format);
  234. return 0;
  235. }
  236. static void __exit exit_som_binfmt(void)
  237. {
  238. /* Remove the SOM loader. */
  239. unregister_binfmt(&som_format);
  240. }
  241. core_initcall(init_som_binfmt);
  242. module_exit(exit_som_binfmt);
  243. MODULE_LICENSE("GPL");