diff options
-rw-r--r-- | shell/ash.c | 68 |
1 files changed, 68 insertions, 0 deletions
diff --git a/shell/ash.c b/shell/ash.c index 3dfe30006..2e47fd272 100644 --- a/shell/ash.c +++ b/shell/ash.c | |||
@@ -13179,6 +13179,7 @@ static const forkpoint_fn forkpoints[] = { | |||
13179 | }; | 13179 | }; |
13180 | 13180 | ||
13181 | static struct forkshell* forkshell_prepare(struct forkshell *fs); | 13181 | static struct forkshell* forkshell_prepare(struct forkshell *fs); |
13182 | static void forkshell_init(const char *idstr); | ||
13182 | #endif | 13183 | #endif |
13183 | 13184 | ||
13184 | /* | 13185 | /* |
@@ -13248,6 +13249,15 @@ int ash_main(int argc UNUSED_PARAM, char **argv) | |||
13248 | 13249 | ||
13249 | init(); | 13250 | init(); |
13250 | setstackmark(&smark); | 13251 | setstackmark(&smark); |
13252 | |||
13253 | #if ENABLE_PLATFORM_MINGW32 | ||
13254 | if (argc == 3 && !strcmp(argv[1], "--forkshell")) { | ||
13255 | forkshell_init(argv[2]); | ||
13256 | |||
13257 | /* NOTREACHED */ | ||
13258 | bb_error_msg_and_die("subshell ended unexpectedly"); | ||
13259 | } | ||
13260 | #endif | ||
13251 | procargs(argv); | 13261 | procargs(argv); |
13252 | 13262 | ||
13253 | #if ENABLE_FEATURE_EDITING_SAVEHISTORY | 13263 | #if ENABLE_FEATURE_EDITING_SAVEHISTORY |
@@ -13770,6 +13780,64 @@ forkshell_prepare(struct forkshell *fs) | |||
13770 | new->hMapFile = h; | 13780 | new->hMapFile = h; |
13771 | return new; | 13781 | return new; |
13772 | } | 13782 | } |
13783 | |||
13784 | #undef exception_handler | ||
13785 | #undef trap | ||
13786 | #undef trap_ptr | ||
13787 | static void | ||
13788 | forkshell_init(const char *idstr) | ||
13789 | { | ||
13790 | struct forkshell *fs; | ||
13791 | int map_handle; | ||
13792 | HANDLE h; | ||
13793 | struct globals_var **gvpp; | ||
13794 | struct globals_misc **gmpp; | ||
13795 | int i; | ||
13796 | |||
13797 | if (sscanf(idstr, "%x", &map_handle) != 1) | ||
13798 | bb_error_msg_and_die("invalid forkshell ID"); | ||
13799 | |||
13800 | h = (HANDLE)map_handle; | ||
13801 | fs = (struct forkshell *)MapViewOfFile(h, FILE_MAP_WRITE, 0,0, 0); | ||
13802 | if (!fs) | ||
13803 | bb_error_msg_and_die("Invalid forkshell memory"); | ||
13804 | |||
13805 | /* pointer fixup */ | ||
13806 | nodeptr = (int*)((char*)fs + fs->nodeptr_offset); | ||
13807 | while (*nodeptr) { | ||
13808 | int *ptr = (int*)((char*)fs + (*nodeptr - (int)fs->old_base)); | ||
13809 | if (*ptr) | ||
13810 | *ptr -= ((int)fs->old_base - (int)fs); | ||
13811 | nodeptr++; | ||
13812 | } | ||
13813 | /* Now fix up stuff that can't be transferred */ | ||
13814 | fs->fp = forkpoints[fs->fpid]; | ||
13815 | for (i = 0; i < ARRAY_SIZE(varinit_data); i++) | ||
13816 | fs->gvp->varinit[i].func = varinit_data[i].func; | ||
13817 | for (i = 0; i < CMDTABLESIZE; i++) { | ||
13818 | struct tblentry *e = fs->cmdtable[i]; | ||
13819 | while (e) { | ||
13820 | if (e->cmdtype == CMDBUILTIN) | ||
13821 | e->param.cmd = builtintab + (int)e->param.cmd; | ||
13822 | e = e->next; | ||
13823 | } | ||
13824 | } | ||
13825 | fs->gmp->exception_handler = ash_ptr_to_globals_misc->exception_handler; | ||
13826 | for (i = 0; i < NSIG; i++) | ||
13827 | fs->gmp->trap[i] = ash_ptr_to_globals_misc->trap[i]; | ||
13828 | fs->gmp->trap_ptr = ash_ptr_to_globals_misc->trap_ptr; | ||
13829 | |||
13830 | /* Switch global variables */ | ||
13831 | gvpp = (struct globals_var **)&ash_ptr_to_globals_var; | ||
13832 | *gvpp = fs->gvp; | ||
13833 | gmpp = (struct globals_misc **)&ash_ptr_to_globals_misc; | ||
13834 | *gmpp = fs->gmp; | ||
13835 | localvars = fs->localvars; | ||
13836 | cmdtable = fs->cmdtable; | ||
13837 | |||
13838 | fs->fp(fs); | ||
13839 | } | ||
13840 | |||
13773 | /*- | 13841 | /*- |
13774 | * Copyright (c) 1989, 1991, 1993, 1994 | 13842 | * Copyright (c) 1989, 1991, 1993, 1994 |
13775 | * The Regents of the University of California. All rights reserved. | 13843 | * The Regents of the University of California. All rights reserved. |