1 /*
2 * Copyright 2010 Louis Lenders
3 * Copyright 2012 Hans Leidekker for CodeWeavers
4 *
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
9 *
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
14 *
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
18 */
19
20 #define COBJMACROS
21
22 #include <stdio.h>
23 #include "windows.h"
24 #include "ocidl.h"
25 #include "initguid.h"
26 #include "objidl.h"
27 #include "wbemcli.h"
28 #include "wmic.h"
29
30 #include "wine/debug.h"
31 #include "wine/unicode.h"
32
33 WINE_DEFAULT_DEBUG_CHANNEL(wmic);
34
35 static const WCHAR biosW[] =
36 {'b','i','o','s',0};
37 static const WCHAR computersystemW[] =
38 {'c','o','m','p','u','t','e','r','s','y','s','t','e','m',0};
39 static const WCHAR cpuW[] =
40 {'c','p','u',0};
41 static const WCHAR logicaldiskW[] =
42 {'L','o','g','i','c','a','l','D','i','s','k',0};
43 static const WCHAR nicW[] =
44 {'n','i','c',0};
45 static const WCHAR osW[] =
46 {'o','s',0};
47 static const WCHAR processW[] =
48 {'p','r','o','c','e','s','s',0};
49
50 static const WCHAR win32_biosW[] =
51 {'W','i','n','3','2','_','B','I','O','S',0};
52 static const WCHAR win32_computersystemW[] =
53 {'W','i','n','3','2','_','C','o','m','p','u','t','e','r','S','y','s','t','e','m',0};
54 static const WCHAR win32_logicaldiskW[] =
55 {'W','i','n','3','2','_','L','o','g','i','c','a','l','D','i','s','k',0};
56 static const WCHAR win32_networkadapterW[] =
57 {'W','i','n','3','2','_','N','e','t','w','o','r','k','A','d','a','p','t','e','r',0};
58 static const WCHAR win32_operatingsystemW[] =
59 {'W','i','n','3','2','_','O','p','e','r','a','t','i','n','g','S','y','s','t','e','m',0};
60 static const WCHAR win32_processW[] =
61 {'W','i','n','3','2','_','P','r','o','c','e','s','s',0};
62 static const WCHAR win32_processorW[] =
63 {'W','i','n','3','2','_','P','r','o','c','e','s','s','o','r',0};
64
65 static const struct
66 {
67 const WCHAR *alias;
68 const WCHAR *class;
69 }
70 alias_map[] =
71 {
72 { biosW, win32_biosW },
73 { computersystemW, win32_computersystemW },
74 { cpuW, win32_processorW },
75 { logicaldiskW, win32_logicaldiskW },
76 { nicW, win32_networkadapterW },
77 { osW, win32_operatingsystemW },
78 { processW, win32_processW }
79 };
80
find_class(const WCHAR * alias)81 static const WCHAR *find_class( const WCHAR *alias )
82 {
83 unsigned int i;
84
85 for (i = 0; i < ARRAY_SIZE(alias_map); i++)
86 {
87 if (!strcmpiW( alias, alias_map[i].alias )) return alias_map[i].class;
88 }
89 return NULL;
90 }
91
strdupW(const WCHAR * src)92 static inline WCHAR *strdupW( const WCHAR *src )
93 {
94 WCHAR *dst;
95 if (!src) return NULL;
96 if (!(dst = HeapAlloc( GetProcessHeap(), 0, (strlenW( src ) + 1) * sizeof(WCHAR) ))) return NULL;
97 strcpyW( dst, src );
98 return dst;
99 }
100
find_prop(IWbemClassObject * class,const WCHAR * prop)101 static WCHAR *find_prop( IWbemClassObject *class, const WCHAR *prop )
102 {
103 SAFEARRAY *sa;
104 WCHAR *ret = NULL;
105 LONG i, last_index = 0;
106 BSTR str;
107
108 if (IWbemClassObject_GetNames( class, NULL, WBEM_FLAG_ALWAYS, NULL, &sa ) != S_OK) return NULL;
109
110 SafeArrayGetUBound( sa, 1, &last_index );
111 for (i = 0; i <= last_index; i++)
112 {
113 SafeArrayGetElement( sa, &i, &str );
114 if (!strcmpiW( str, prop ))
115 {
116 ret = strdupW( str );
117 break;
118 }
119 }
120 SafeArrayDestroy( sa );
121 return ret;
122 }
123
output_string(HANDLE handle,const WCHAR * msg,...)124 static int output_string( HANDLE handle, const WCHAR *msg, ... )
125 {
126 va_list va_args;
127 int len;
128 DWORD count;
129 WCHAR buffer[8192];
130
131 va_start( va_args, msg );
132 len = vsnprintfW( buffer, ARRAY_SIZE(buffer), msg, va_args );
133 va_end( va_args );
134
135 if (!WriteConsoleW( handle, buffer, len, &count, NULL ))
136 WriteFile( handle, buffer, len * sizeof(WCHAR), &count, FALSE );
137
138 return count;
139 }
140
output_error(int msg)141 static int output_error( int msg )
142 {
143 static const WCHAR fmtW[] = {'%','s',0};
144 WCHAR buffer[8192];
145
146 LoadStringW( GetModuleHandleW(NULL), msg, buffer, ARRAY_SIZE(buffer));
147 return output_string( GetStdHandle(STD_ERROR_HANDLE), fmtW, buffer );
148 }
149
output_header(const WCHAR * prop,ULONG column_width)150 static int output_header( const WCHAR *prop, ULONG column_width )
151 {
152 static const WCHAR bomW[] = {0xfeff}, fmtW[] = {'%','-','*','s','\r','\n',0};
153 int len;
154 DWORD count;
155 WCHAR buffer[8192];
156
157 len = snprintfW( buffer, ARRAY_SIZE(buffer), fmtW, column_width, prop );
158
159 if (!WriteConsoleW( GetStdHandle(STD_OUTPUT_HANDLE), buffer, len, &count, NULL )) /* redirected */
160 {
161 WriteFile( GetStdHandle(STD_OUTPUT_HANDLE), bomW, sizeof(bomW), &count, FALSE );
162 WriteFile( GetStdHandle(STD_OUTPUT_HANDLE), buffer, len * sizeof(WCHAR), &count, FALSE );
163 count += sizeof(bomW);
164 }
165
166 return count;
167 }
168
output_line(const WCHAR * str,ULONG column_width)169 static int output_line( const WCHAR *str, ULONG column_width )
170 {
171 static const WCHAR fmtW[] = {'%','-','*','s','\r','\n',0};
172 return output_string( GetStdHandle(STD_OUTPUT_HANDLE), fmtW, column_width, str );
173 }
174
query_prop(const WCHAR * class,const WCHAR * propname)175 static int query_prop( const WCHAR *class, const WCHAR *propname )
176 {
177 static const WCHAR select_allW[] = {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',0};
178 static const WCHAR cimv2W[] = {'R','O','O','T','\\','C','I','M','V','2',0};
179 static const WCHAR wqlW[] = {'W','Q','L',0};
180 HRESULT hr;
181 IWbemLocator *locator = NULL;
182 IWbemServices *services = NULL;
183 IEnumWbemClassObject *result = NULL;
184 LONG flags = WBEM_FLAG_RETURN_IMMEDIATELY | WBEM_FLAG_FORWARD_ONLY;
185 BSTR path = NULL, wql = NULL, query = NULL;
186 WCHAR *prop = NULL;
187 BOOL first = TRUE;
188 int len, ret = -1;
189 IWbemClassObject *obj;
190 ULONG count, width = 0;
191 VARIANT v;
192
193 WINE_TRACE("%s, %s\n", debugstr_w(class), debugstr_w(propname));
194
195 CoInitialize( NULL );
196 CoInitializeSecurity( NULL, -1, NULL, NULL, RPC_C_AUTHN_LEVEL_DEFAULT,
197 RPC_C_IMP_LEVEL_IMPERSONATE, NULL, EOAC_NONE, NULL );
198
199 hr = CoCreateInstance( &CLSID_WbemLocator, NULL, CLSCTX_INPROC_SERVER, &IID_IWbemLocator,
200 (void **)&locator );
201 if (hr != S_OK) goto done;
202
203 if (!(path = SysAllocString( cimv2W ))) goto done;
204 hr = IWbemLocator_ConnectServer( locator, path, NULL, NULL, NULL, 0, NULL, NULL, &services );
205 if (hr != S_OK) goto done;
206
207 len = strlenW( class ) + ARRAY_SIZE(select_allW);
208 if (!(query = SysAllocStringLen( NULL, len ))) goto done;
209 strcpyW( query, select_allW );
210 strcatW( query, class );
211
212 if (!(wql = SysAllocString( wqlW ))) goto done;
213 hr = IWbemServices_ExecQuery( services, wql, query, flags, NULL, &result );
214 if (hr != S_OK) goto done;
215
216 for (;;) /* get column width */
217 {
218 IEnumWbemClassObject_Next( result, WBEM_INFINITE, 1, &obj, &count );
219 if (!count) break;
220
221 if (!prop && !(prop = find_prop( obj, propname )))
222 {
223 output_error( STRING_INVALID_QUERY );
224 goto done;
225 }
226 if (IWbemClassObject_Get( obj, prop, 0, &v, NULL, NULL ) == WBEM_S_NO_ERROR)
227 {
228 VariantChangeType( &v, &v, 0, VT_BSTR );
229 width = max( strlenW( V_BSTR( &v ) ), width );
230 VariantClear( &v );
231 }
232 IWbemClassObject_Release( obj );
233 }
234 width += 2;
235
236 IEnumWbemClassObject_Reset( result );
237 for (;;)
238 {
239 IEnumWbemClassObject_Next( result, WBEM_INFINITE, 1, &obj, &count );
240 if (!count) break;
241
242 if (first)
243 {
244 output_header( prop, width );
245 first = FALSE;
246 }
247 if (IWbemClassObject_Get( obj, prop, 0, &v, NULL, NULL ) == WBEM_S_NO_ERROR)
248 {
249 VariantChangeType( &v, &v, 0, VT_BSTR );
250 output_line( V_BSTR( &v ), width );
251 VariantClear( &v );
252 }
253 IWbemClassObject_Release( obj );
254 }
255 ret = 0;
256
257 done:
258 if (result) IEnumWbemClassObject_Release( result );
259 if (services) IWbemServices_Release( services );
260 if (locator) IWbemLocator_Release( locator );
261 SysFreeString( path );
262 SysFreeString( query );
263 SysFreeString( wql );
264 HeapFree( GetProcessHeap(), 0, prop );
265 CoUninitialize();
266 return ret;
267 }
268
wmain(int argc,WCHAR * argv[])269 int wmain(int argc, WCHAR *argv[])
270 {
271 static const WCHAR getW[] = {'g','e','t',0};
272 static const WCHAR quitW[] = {'q','u','i','t',0};
273 static const WCHAR exitW[] = {'e','x','i','t',0};
274 static const WCHAR pathW[] = {'p','a','t','h',0};
275 static const WCHAR classW[] = {'c','l','a','s','s',0};
276 static const WCHAR contextW[] = {'c','o','n','t','e','x','t',0};
277 const WCHAR *class, *value;
278 int i;
279
280 for (i = 1; i < argc && argv[i][0] == '/'; i++)
281 WINE_FIXME( "command line switch %s not supported\n", debugstr_w(argv[i]) );
282
283 if (i >= argc)
284 goto not_supported;
285
286 if (!strcmpiW( argv[i], quitW ) ||
287 !strcmpiW( argv[i], exitW ))
288 {
289 return 0;
290 }
291
292 if (!strcmpiW( argv[i], classW) ||
293 !strcmpiW( argv[i], contextW ))
294 {
295 WINE_FIXME( "command %s not supported\n", debugstr_w(argv[i]) );
296 goto not_supported;
297 }
298
299 if (!strcmpiW( argv[i], pathW ))
300 {
301 if (++i >= argc)
302 {
303 output_error( STRING_INVALID_PATH );
304 return 1;
305 }
306 class = argv[i];
307 }
308 else
309 {
310 class = find_class( argv[i] );
311 if (!class)
312 {
313 output_error( STRING_ALIAS_NOT_FOUND );
314 return 1;
315 }
316 }
317
318 if (++i >= argc)
319 goto not_supported;
320
321 if (!strcmpiW( argv[i], getW ))
322 {
323 if (++i >= argc)
324 goto not_supported;
325 value = argv[i];
326 return query_prop( class, value );
327 }
328
329 not_supported:
330 output_error( STRING_CMDLINE_NOT_SUPPORTED );
331 return 1;
332 }
333