Do not print 'wine: no executeable found' if we did execute something.
[wine] / miscemu / main.c
1 /*
2  * Emulator initialisation code
3  *
4  */
5
6 #include <assert.h>
7 #include "callback.h"
8 #include "debug.h"
9 #include "debugger.h"
10 #include "main.h"
11 #include "miscemu.h"
12 #include "module.h"
13 #include "options.h"
14 #include "process.h"
15 #include "win16drv.h"
16 #include "thread.h"
17 #include "task.h"
18 #include "stackframe.h"
19 #include "windows.h"
20
21 static int MAIN_argc;
22 static char **MAIN_argv;
23
24 /***********************************************************************
25  *           Emulator initialisation
26  */
27 BOOL32 MAIN_EmulatorInit(void)
28 {
29     /* Main initialization */
30     if (!MAIN_MainInit()) return FALSE;
31
32     /* Initialize relay code */
33     if (!RELAY_Init()) return FALSE;
34
35     /* Create the Win16 printer driver */
36     if (!WIN16DRV_Init()) return FALSE;
37
38     return TRUE;
39 }
40
41
42 /***********************************************************************
43  *           Main loop of initial task
44  */
45 void MAIN_EmulatorRun( void )
46 {
47     extern void THUNK_InitCallout( void );
48     char startProg[256], defProg[256];
49     HINSTANCE32 handle;
50     int i, tasks = 0;
51     MSG32 msg;
52
53     /* Load system DLLs into the initial process (and initialize them) */
54     if (   !LoadLibrary16("GDI.EXE" ) || !LoadLibrary32A("GDI32.DLL" )
55         || !LoadLibrary16("USER.EXE") || !LoadLibrary32A("USER32.DLL"))
56         ExitProcess( 1 );
57
58     /* Get pointers to USER routines called by KERNEL */
59     THUNK_InitCallout();
60
61     /* Add the Default Program if no program on the command line */
62     if (!MAIN_argv[1])
63     {
64         PROFILE_GetWineIniString( "programs", "Default", "",
65                                   defProg, sizeof(defProg) );
66         if (defProg[0]) MAIN_argv[MAIN_argc++] = defProg;
67     }
68     
69     /* Add the Startup Program to the run list */
70     PROFILE_GetWineIniString( "programs", "Startup", "", 
71                                startProg, sizeof(startProg) );
72     if (startProg[0]) MAIN_argv[MAIN_argc++] = startProg;
73
74     /* Abort if no executable on command line */
75     if (MAIN_argc <= 1) 
76     {
77         MAIN_Usage(MAIN_argv[0]);
78         exit(1);
79     }
80
81     /* Load and run executables given on command line */
82     for (i = 1; i < MAIN_argc; i++)
83     {
84         if ((handle = WinExec32( MAIN_argv[i], SW_SHOWNORMAL )) < 32)
85         {
86             MSG("wine: can't exec '%s': ", MAIN_argv[i]);
87             switch (handle)
88             {
89             case 2: MSG("file not found\n" ); break;
90             case 11: MSG("invalid exe file\n" ); break;
91             default: MSG("error=%d\n", handle ); break;
92             }
93         }
94         else tasks++;
95     }
96
97     if (!tasks)
98     {
99         MSG("wine: no executable file found.\n" );
100         ExitProcess( 0 );
101     }
102
103
104     /* Start message loop for desktop window */
105
106     while ( GetNumTasks() > 1 && Callout.GetMessage32A( &msg, 0, 0, 0 ) )
107     {
108         Callout.TranslateMessage32( &msg );
109         Callout.DispatchMessage32A( &msg );
110     }
111
112     ExitProcess( 0 );
113 }
114
115
116 /**********************************************************************
117  *           main
118  */
119 int main( int argc, char *argv[] )
120 {
121     NE_MODULE *pModule;
122     HINSTANCE16 hInstance;
123     extern char * DEBUG_argv0;
124
125     __winelib = 0;  /* First of all, clear the Winelib flag */
126
127     /*
128      * Save this so that the internal debugger can get a hold of it if
129      * it needs to.
130      */
131     DEBUG_argv0 = argv[0];
132
133     /* Create the initial process */
134     if (!PROCESS_Init()) return FALSE;
135
136     /* Parse command-line */
137     if (!MAIN_WineInit( &argc, argv )) return 1;
138     MAIN_argc = argc; MAIN_argv = argv;
139
140     /* Handle -dll option (hack) */
141     if (Options.dllFlags)
142     {
143         if (!BUILTIN_ParseDLLOptions( Options.dllFlags ))
144         {
145             MSG("%s: Syntax: -dll +xxx,... or -dll -xxx,...\n",
146                      argv[0] );
147             BUILTIN_PrintDLLs();
148             exit(1);
149         }
150     }
151
152     /* Set up debugger/instruction emulation callback routines */
153     ctx_debug_call              = ctx_debug;
154     fnWINE_Debugger             = wine_debug;
155     fnINSTR_EmulateInstruction  = INSTR_EmulateInstruction;
156
157     if (Options.debug) 
158         TASK_AddTaskEntryBreakpoint = DEBUG_AddTaskEntryBreakpoint;
159
160     /* Initialize everything */
161     if (!MAIN_EmulatorInit()) return 1;
162
163     /* Load kernel modules */
164     if (!LoadLibrary16(  "KERNEL" )) return 1;
165     if (!LoadLibrary32A( "KERNEL32" )) return 1;
166
167     /* Create initial task */
168     if ( !(pModule = NE_GetPtr( GetModuleHandle16( "KERNEL32" ) )) ) return 1;
169     hInstance = NE_CreateInstance( pModule, NULL, TRUE );
170     PROCESS_Current()->task = TASK_Create( THREAD_Current(), pModule, hInstance, 0, FALSE );
171
172     /* Initialize CALL32 routines */
173     /* This needs to be done just before switching stacks */
174     IF1632_CallLargeStack = (int (*)(int (*func)(), void *arg))CALL32_Init();
175
176     /* Switch to initial task */
177     CURRENT_STACK16->frame32->retaddr = (DWORD)MAIN_EmulatorRun;
178     TASK_StartTask( PROCESS_Current()->task );
179     MSG( "main: Should never happen: returned from TASK_StartTask()\n" );
180     return 0;
181 }