VirtualBox

source: vbox/trunk/src/VBox/Main/cbinding/tstXPCOMCGlue.c@ 16867

Last change on this file since 16867 was 16838, checked in by vboxsync, 16 years ago

Fixed $Revision$ typo.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 11.4 KB
Line 
1/* $Revision: 16838 $ */
2/** @file tstXPCOMCGlue.c
3 * Demonstrator program to illustrate use of C bindings of Main API.
4 *
5 * Linux only at the moment due to shared library magic in the Makefile.
6 */
7
8/*
9 * Copyright (C) 2009 Sun Microsystems, Inc.
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 *
19 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
20 * Clara, CA 95054 USA or visit http://www.sun.com if you need
21 * additional information or have any questions.
22 */
23
24/*******************************************************************************
25* Header Files *
26*******************************************************************************/
27#include "VBoxXPCOMCGlue.h"
28#include <stdio.h>
29#include <string.h>
30#include <stdlib.h>
31#include <sys/stat.h>
32
33static char *nsIDToString(nsID *guid);
34static void listVMs(IVirtualBox *virtualBox, ISession *session);
35static void startVM(IVirtualBox *virtualBox, ISession *session, nsID *id);
36
37/**
38 * Helper function to convert an nsID into a human readable string.
39 *
40 * @returns result string, allocated. Has to be freed using free()
41 * @param guid Pointer to nsID that will be converted.
42 */
43static char *nsIDToString(nsID *guid)
44{
45 /* Avoid magic number 39. Yes, sizeof "literal" includes the NUL byte. */
46 char *res = malloc(sizeof "{12345678-1234-1234-1234-123456789012}");
47
48 if (res != NULL)
49 {
50 sprintf(res, "{%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x}",
51 (unsigned)guid->m0, (unsigned)guid->m1, (unsigned)guid->m2,
52 (unsigned)guid->m3[0], (unsigned)guid->m3[1],
53 (unsigned)guid->m3[2], (unsigned)guid->m3[3],
54 (unsigned)guid->m3[4], (unsigned)guid->m3[5],
55 (unsigned)guid->m3[6], (unsigned)guid->m3[7]);
56 }
57 return res;
58}
59
60/**
61 * List the registered VMs.
62 *
63 * @param virtualBox ptr to IVirtualBox object
64 * @param session ptr to ISession object
65 */
66static void listVMs(IVirtualBox *virtualBox, ISession *session)
67{
68 nsresult rc;
69 IMachine **machines = NULL;
70 PRUint32 machineCnt = 0;
71 PRUint32 i;
72 unsigned start_id;
73
74 /*
75 * Get the list of all registered VMs.
76 */
77
78 rc = virtualBox->vtbl->GetMachines2(virtualBox, &machineCnt, &machines);
79 if (NS_FAILED(rc))
80 {
81 fprintf(stderr, "could not get list of machines, rc=%08x\n",
82 (unsigned)rc);
83 return;
84 }
85
86 if (machineCnt == 0)
87 {
88 printf("\tNo VMs\n");
89 return;
90 }
91
92 printf("VM List:\n\n");
93
94 /*
95 * Iterate through the collection.
96 */
97
98 for (i = 0; i < machineCnt; ++i)
99 {
100 IMachine *machine = machines[i];
101 PRBool isAccessible = PR_FALSE;
102
103 printf("\tMachine #%u\n", (unsigned)i);
104
105 if (!machine)
106 {
107 printf("\t(skipped, NULL)\n");
108 continue;
109 }
110
111 machine->vtbl->GetAccessible(machine, &isAccessible);
112
113 if (isAccessible)
114 {
115 PRUnichar *machineNameUtf16;
116 char *machineName;
117
118 machine->vtbl->GetName(machine, &machineNameUtf16);
119 g_pVBoxFuncs->pfnUtf16ToUtf8(machineNameUtf16,&machineName);
120 printf("\tName: %s\n", machineName);
121
122 g_pVBoxFuncs->pfnUtf8Free(machineName);
123 g_pVBoxFuncs->pfnComUnallocMem(machineNameUtf16);
124 }
125 else
126 {
127 printf("\tName: <inaccessible>\n");
128 }
129
130
131 {
132 nsID *iid = NULL;
133 char *uuidString;
134
135 machine->vtbl->GetId(machine, &iid);
136 uuidString = nsIDToString(iid);
137 printf("\tUUID: %s\n", uuidString);
138
139 free(uuidString);
140 g_pVBoxFuncs->pfnComUnallocMem(iid);
141 }
142
143 if (isAccessible)
144 {
145 {
146 PRUnichar *configFile;
147 char *configFile1 = calloc((size_t)64, (size_t)1);
148
149 machine->vtbl->GetSettingsFilePath(machine, &configFile);
150 g_pVBoxFuncs->pfnUtf16ToUtf8(configFile, &configFile1);
151 printf("\tConfig file: %s\n", configFile1);
152
153 free(configFile1);
154 g_pVBoxFuncs->pfnComUnallocMem(configFile);
155 }
156
157 {
158 PRUint32 memorySize;
159
160 machine->vtbl->GetMemorySize(machine, &memorySize);
161 printf("\tMemory size: %uMB\n", memorySize);
162 }
163
164 {
165 PRUnichar *typeId;
166 PRUnichar *osNameUtf16;
167 char *osName;
168 IGuestOSType *osType = NULL;
169
170 machine->vtbl->GetOSTypeId(machine, &typeId);
171 virtualBox->vtbl->GetGuestOSType(virtualBox, typeId, &osType);
172 osType->vtbl->GetDescription(osType, &osNameUtf16);
173 g_pVBoxFuncs->pfnUtf16ToUtf8(osNameUtf16,&osName);
174 printf("\tGuest OS: %s\n\n", osName);
175
176 osType->vtbl->nsisupports.Release((void *)osType);
177 g_pVBoxFuncs->pfnUtf8Free(osName);
178 g_pVBoxFuncs->pfnComUnallocMem(osNameUtf16);
179 g_pVBoxFuncs->pfnComUnallocMem(typeId);
180 }
181 }
182 }
183
184 /*
185 * Let the user chose a machine to start.
186 */
187
188 printf("Type Machine# to start (0 - %u) or 'quit' to do nothing: ",
189 (unsigned)(machineCnt - 1));
190 fflush(stdout);
191
192 if (scanf("%u", &start_id) == 1 && start_id < machineCnt)
193 {
194 IMachine *machine = machines[start_id];
195
196 if (machine)
197 {
198 nsID *iid = NULL;
199
200 machine->vtbl->GetId(machine, &iid);
201 startVM(virtualBox, session, iid);
202
203 g_pVBoxFuncs->pfnComUnallocMem(iid);
204 }
205 }
206
207 /*
208 * Don't forget to release the objects in the array.
209 */
210
211 for (i = 0; i < machineCnt; ++i)
212 {
213 IMachine *machine = machines[i];
214
215 if (machine)
216 {
217 machine->vtbl->nsisupports.Release((void *)machine);
218 }
219 }
220}
221
222/**
223 * Start a VM.
224 *
225 * @param virtualBox ptr to IVirtualBox object
226 * @param session ptr to ISession object
227 * @param id identifies the machine to start
228 */
229static void startVM(IVirtualBox *virtualBox, ISession *session, nsID *id)
230{
231 nsresult rc;
232 IMachine *machine = NULL;
233 IProgress *progress = NULL;
234 PRUnichar *env = NULL;
235 PRUnichar *sessionType;
236
237 rc = virtualBox->vtbl->GetMachine(virtualBox, id, &machine);
238
239 if (NS_FAILED(rc) || !machine)
240 {
241 fprintf(stderr, "Error: Couldn't get the machine handle.\n");
242 return;
243 }
244
245 g_pVBoxFuncs->pfnUtf8ToUtf16("gui", &sessionType);
246
247 rc = virtualBox->vtbl->OpenRemoteSession(
248 virtualBox,
249 session,
250 id,
251 sessionType,
252 env,
253 &progress
254 );
255
256 g_pVBoxFuncs->pfnUtf16Free(sessionType);
257
258 if (NS_FAILED(rc))
259 {
260 fprintf(stderr, "Error: OpenRemoteSession failed.\n");
261 }
262 else
263 {
264 PRBool completed;
265 nsresult resultCode;
266
267 printf("Waiting for the remote session to open...\n");
268 progress->vtbl->WaitForCompletion(progress, -1);
269
270 rc = progress->vtbl->GetCompleted(progress, &completed);
271 if (NS_FAILED(rc))
272 {
273 fprintf (stderr, "Error: GetCompleted status failed.\n");
274 }
275
276 progress->vtbl->GetResultCode(progress, &resultCode);
277 if (NS_FAILED(resultCode))
278 {
279 IVirtualBoxErrorInfo *errorInfo;
280 PRUnichar *textUtf16;
281 char *text;
282
283 progress->vtbl->GetErrorInfo(progress, &errorInfo);
284 errorInfo->vtbl->GetText(errorInfo, &textUtf16);
285 g_pVBoxFuncs->pfnUtf16ToUtf8(textUtf16, &text);
286 printf("Error: %s\n", text);
287
288 g_pVBoxFuncs->pfnComUnallocMem(textUtf16);
289 g_pVBoxFuncs->pfnUtf8Free(text);
290 }
291 else
292 {
293 fprintf(stderr, "Remote session has been successfully opened.\n");
294 }
295 progress->vtbl->nsisupports.Release((void *)progress);
296 }
297
298 /* It's important to always release resources. */
299 machine->vtbl->nsisupports.Release((void *)machine);
300}
301
302/* Main - Start the ball rolling. */
303
304int main(int argc, char **argv)
305{
306 IVirtualBox *vbox = NULL;
307 ISession *session = NULL;
308 PRUint32 revision = 0;
309 PRUnichar *versionUtf16 = NULL;
310 PRUnichar *homefolderUtf16 = NULL;
311 struct stat stIgnored;
312 nsresult rc; /* Result code of various function (method) calls. */
313
314 printf("Starting Main\n");
315
316 /*
317 * VBoxComInitialize does all the necessary startup action and
318 * provides us with pointers to vbox and session handles.
319 * It should be matched by a call to VBoxComUninitialize(vbox)
320 * when done.
321 */
322
323 if (VBoxCGlueInit(NULL) != 0)
324 {
325 fprintf(stderr, "%s: FATAL: VBoxCGlueInit failed: %s\n",
326 argv[0], g_szVBoxErrMsg);
327 return EXIT_FAILURE;
328 }
329
330 g_pVBoxFuncs->pfnComInitialize(&vbox, &session);
331 if (vbox == NULL)
332 {
333 fprintf(stderr, "%s: FATAL: could not get vbox handle\n", argv[0]);
334 return EXIT_FAILURE;
335 }
336 if (session == NULL)
337 {
338 fprintf(stderr, "%s: FATAL: could not get session handle\n", argv[0]);
339 return EXIT_FAILURE;
340 }
341
342 /*
343 * Now ask for revision, version and home folder information of
344 * this vbox. Were not using fancy macros here so it
345 * remains easy to see how we access C++'s vtable.
346 */
347
348 printf("----------------------------------------------------\n");
349
350 /* 1. Revision */
351
352 rc = vbox->vtbl->GetRevision(vbox, &revision);
353 if (NS_SUCCEEDED(rc))
354 {
355 printf("\tRevision: %u\n", revision);
356 }
357 else
358 {
359 fprintf(stderr, "%s: GetRevision() returned %08x\n",
360 argv[0], (unsigned)rc);
361 }
362
363 /* 2. Version */
364
365 rc = vbox->vtbl->GetVersion(vbox, &versionUtf16);
366 if (NS_SUCCEEDED(rc))
367 {
368 char *version = NULL;
369 g_pVBoxFuncs->pfnUtf16ToUtf8(versionUtf16, &version);
370 printf("\tVersion: %s\n", version);
371 g_pVBoxFuncs->pfnUtf8Free(version);
372 g_pVBoxFuncs->pfnComUnallocMem(versionUtf16);
373 }
374 else
375 {
376 fprintf(stderr, "%s: GetVersion() returned %08x\n",
377 argv[0], (unsigned)rc);
378 }
379
380 /* 3. Home Folder */
381
382 rc = vbox->vtbl->GetHomeFolder(vbox, &homefolderUtf16);
383 if (NS_SUCCEEDED(rc))
384 {
385 char *homefolder = NULL;
386 g_pVBoxFuncs->pfnUtf16ToUtf8(homefolderUtf16, &homefolder);
387 printf("\tHomeFolder: %s\n", homefolder);
388 g_pVBoxFuncs->pfnUtf8Free(homefolder);
389 g_pVBoxFuncs->pfnComUnallocMem(homefolderUtf16);
390 }
391 else
392 {
393 fprintf(stderr, "%s: GetHomeFolder() returned %08x\n",
394 argv[0], (unsigned)rc);
395 }
396
397 listVMs(vbox, session);
398 session->vtbl->Close(session);
399
400 printf("----------------------------------------------------\n");
401
402 /*
403 * Do as mom told us: always clean up after yourself.
404 */
405
406 g_pVBoxFuncs->pfnComUninitialize();
407 VBoxCGlueTerm();
408 printf("Finished Main\n");
409
410 return 0;
411}
412/* vim: set ts=4 sw=4 et: */
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette