blob: 3353f36627ac8e1b0286c5a9f7653308fde13df4 [file] [log] [blame]
Denis Vlasenko41f5add2007-11-28 06:49:42 +00001/* vi: set sw=4 ts=4: */
2/*
3 * Applet table generator.
4 * Runs on host and produces include/applet_tables.h
5 *
6 * Copyright (C) 2007 Denys Vlasenko <vda.linux@googlemail.com>
7 *
8 * Licensed under GPLv2, see file License in this tarball for details.
9 */
10
11#include <stdlib.h>
12#include <string.h>
13#include <stdio.h>
14
15#include "../include/autoconf.h"
16#include "../include/busybox.h"
17
18struct bb_applet {
19 const char *name;
20 const char *main;
21 enum bb_install_loc_t install_loc;
22 enum bb_suid_t need_suid;
23 /* true if instead of fork(); exec("applet"); waitpid();
24 * one can do fork(); exit(applet_main(argc,argv)); waitpid(); */
25 unsigned char noexec;
26 /* Even nicer */
27 /* true if instead of fork(); exec("applet"); waitpid();
28 * one can simply call applet_main(argc,argv); */
29 unsigned char nofork;
30};
31
32/* Define struct bb_applet applets[] */
33#include "../include/applets.h"
34
35enum { NUM_APPLETS = sizeof(applets)/sizeof(applets[0]) };
36
37static int offset[NUM_APPLETS];
38
39static int cmp_name(const void *a, const void *b)
40{
41 const struct bb_applet *aa = a;
42 const struct bb_applet *bb = b;
43 return strcmp(aa->name, bb->name);
44}
45
46int main(int argc, char **argv)
47{
48 int i;
49 int ofs;
50
51 qsort(applets, NUM_APPLETS, sizeof(applets[0]), cmp_name);
52
Denis Vlasenko41f5add2007-11-28 06:49:42 +000053 ofs = 0;
Denis Vlasenko41f5add2007-11-28 06:49:42 +000054 for (i = 0; i < NUM_APPLETS; i++) {
55 offset[i] = ofs;
56 ofs += strlen(applets[i].name) + 1;
Denis Vlasenko745cd172007-11-29 03:31:20 +000057 }
58 /* We reuse 4 high-order bits of offset array for other purposes,
59 * so if they are indeed needed, refuse to proceed */
60 if (ofs > 0xfff)
61 return 1;
62 if (!argv[1])
63 return 1;
64
65 i = open(argv[1], O_WRONLY | O_TRUNC | O_CREAT, 0666);
66 if (i < 0)
67 return 1;
Denis Vlasenkof7be20e2007-12-24 14:09:19 +000068 dup2(i, 1);
Denis Vlasenko745cd172007-11-29 03:31:20 +000069
70 /* Keep in sync with include/busybox.h! */
71
72 puts("/* This is a generated file, don't edit */");
73
Denis Vlasenko5fee2e12008-01-05 03:26:41 +000074 puts("const char applet_names[] ALIGN1 = \"\"\n");
Denis Vlasenko745cd172007-11-29 03:31:20 +000075 for (i = 0; i < NUM_APPLETS; i++) {
Denis Vlasenko41f5add2007-11-28 06:49:42 +000076 printf("\"%s\" \"\\0\"\n", applets[i].name);
77 }
78 puts(";");
79
Denis Vlasenko745cd172007-11-29 03:31:20 +000080 puts("int (*const applet_main[])(int argc, char **argv) = {");
Denis Vlasenko41f5add2007-11-28 06:49:42 +000081 for (i = 0; i < NUM_APPLETS; i++) {
82 printf("%s_main,\n", applets[i].main);
83 }
84 puts("};");
85
Denis Vlasenko41f5add2007-11-28 06:49:42 +000086 puts("const uint16_t applet_nameofs[] ALIGN2 = {");
Denis Vlasenko41f5add2007-11-28 06:49:42 +000087 for (i = 0; i < NUM_APPLETS; i++) {
Denis Vlasenko745cd172007-11-29 03:31:20 +000088 printf("0x%04x,\n",
Denis Vlasenko41f5add2007-11-28 06:49:42 +000089 offset[i]
Denis Vlasenko745cd172007-11-29 03:31:20 +000090#if ENABLE_FEATURE_PREFER_APPLETS
91 + (applets[i].nofork << 12)
92 + (applets[i].noexec << 13)
93#endif
Denis Vlasenko41f5add2007-11-28 06:49:42 +000094#if ENABLE_FEATURE_SUID
95 + (applets[i].need_suid << 14) /* 2 bits */
96#endif
Denis Vlasenko41f5add2007-11-28 06:49:42 +000097 );
98 }
99 puts("};");
100
Denis Vlasenko745cd172007-11-29 03:31:20 +0000101#if ENABLE_FEATURE_INSTALLER
102 puts("const uint8_t applet_install_loc[] ALIGN1 = {");
103 i = 0;
104 while (i < NUM_APPLETS) {
105 int v = applets[i].install_loc; /* 3 bits */
106 if (++i < NUM_APPLETS)
107 v |= applets[i].install_loc << 4; /* 3 bits */
108 printf("0x%02x,\n", v);
109 i++;
110 }
111 puts("};");
112#endif
113
Denis Vlasenko41f5add2007-11-28 06:49:42 +0000114 return 0;
115}