| /* |
| * Copyright 2010 Louis Lenders |
| * Copyright 2012 Hans Leidekker for CodeWeavers |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| |
| #define COBJMACROS |
| |
| #include <stdio.h> |
| #include "windows.h" |
| #include "ocidl.h" |
| #include "initguid.h" |
| #include "objidl.h" |
| #include "wbemcli.h" |
| #include "wmic.h" |
| |
| #include "wine/debug.h" |
| #include "wine/unicode.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(wmic); |
| |
| static const WCHAR biosW[] = |
| {'b','i','o','s',0}; |
| static const WCHAR computersystemW[] = |
| {'c','o','m','p','u','t','e','r','s','y','s','t','e','m',0}; |
| static const WCHAR cpuW[] = |
| {'c','p','u',0}; |
| static const WCHAR logicaldiskW[] = |
| {'L','o','g','i','c','a','l','D','i','s','k',0}; |
| static const WCHAR nicW[] = |
| {'n','i','c',0}; |
| static const WCHAR osW[] = |
| {'o','s',0}; |
| static const WCHAR processW[] = |
| {'p','r','o','c','e','s','s',0}; |
| |
| static const WCHAR win32_biosW[] = |
| {'W','i','n','3','2','_','B','I','O','S',0}; |
| static const WCHAR win32_computersystemW[] = |
| {'W','i','n','3','2','_','C','o','m','p','u','t','e','r','S','y','s','t','e','m',0}; |
| static const WCHAR win32_logicaldiskW[] = |
| {'W','i','n','3','2','_','L','o','g','i','c','a','l','D','i','s','k',0}; |
| static const WCHAR win32_networkadapterW[] = |
| {'W','i','n','3','2','_','N','e','t','w','o','r','k','A','d','a','p','t','e','r',0}; |
| static const WCHAR win32_operatingsystemW[] = |
| {'W','i','n','3','2','_','O','p','e','r','a','t','i','n','g','S','y','s','t','e','m',0}; |
| static const WCHAR win32_processW[] = |
| {'W','i','n','3','2','_','P','r','o','c','e','s','s',0}; |
| static const WCHAR win32_processorW[] = |
| {'W','i','n','3','2','_','P','r','o','c','e','s','s','o','r',0}; |
| |
| static const struct |
| { |
| const WCHAR *alias; |
| const WCHAR *class; |
| } |
| alias_map[] = |
| { |
| { biosW, win32_biosW }, |
| { computersystemW, win32_computersystemW }, |
| { cpuW, win32_processorW }, |
| { logicaldiskW, win32_logicaldiskW }, |
| { nicW, win32_networkadapterW }, |
| { osW, win32_operatingsystemW }, |
| { processW, win32_processW } |
| }; |
| |
| static const WCHAR *find_class( const WCHAR *alias ) |
| { |
| unsigned int i; |
| |
| for (i = 0; i < sizeof(alias_map)/sizeof(alias_map[0]); i++) |
| { |
| if (!strcmpiW( alias, alias_map[i].alias )) return alias_map[i].class; |
| } |
| return NULL; |
| } |
| |
| static inline WCHAR *strdupW( const WCHAR *src ) |
| { |
| WCHAR *dst; |
| if (!src) return NULL; |
| if (!(dst = HeapAlloc( GetProcessHeap(), 0, (strlenW( src ) + 1) * sizeof(WCHAR) ))) return NULL; |
| strcpyW( dst, src ); |
| return dst; |
| } |
| |
| static WCHAR *find_prop( IWbemClassObject *class, const WCHAR *prop ) |
| { |
| SAFEARRAY *sa; |
| WCHAR *ret = NULL; |
| LONG i, last_index = 0; |
| BSTR str; |
| |
| if (IWbemClassObject_GetNames( class, NULL, WBEM_FLAG_ALWAYS, NULL, &sa ) != S_OK) return NULL; |
| |
| SafeArrayGetUBound( sa, 1, &last_index ); |
| for (i = 0; i <= last_index; i++) |
| { |
| SafeArrayGetElement( sa, &i, &str ); |
| if (!strcmpiW( str, prop )) |
| { |
| ret = strdupW( str ); |
| break; |
| } |
| } |
| SafeArrayDestroy( sa ); |
| return ret; |
| } |
| |
| static int output_string( const WCHAR *msg, ... ) |
| { |
| va_list va_args; |
| int wlen; |
| DWORD count, ret; |
| WCHAR buffer[8192]; |
| |
| va_start( va_args, msg ); |
| vsprintfW( buffer, msg, va_args ); |
| va_end( va_args ); |
| |
| wlen = strlenW( buffer ); |
| ret = WriteConsoleW( GetStdHandle(STD_OUTPUT_HANDLE), buffer, wlen, &count, NULL ); |
| if (!ret) |
| { |
| DWORD len; |
| char *msgA; |
| |
| /* On Windows WriteConsoleW() fails if the output is redirected. So fall |
| * back to WriteFile(), assuming the console encoding is still the right |
| * one in that case. |
| */ |
| len = WideCharToMultiByte( GetConsoleOutputCP(), 0, buffer, wlen, NULL, 0, NULL, NULL ); |
| if (!(msgA = HeapAlloc( GetProcessHeap(), 0, len * sizeof(char) ))) return 0; |
| |
| WideCharToMultiByte( GetConsoleOutputCP(), 0, buffer, wlen, msgA, len, NULL, NULL ); |
| WriteFile( GetStdHandle(STD_OUTPUT_HANDLE), msgA, len, &count, FALSE ); |
| HeapFree( GetProcessHeap(), 0, msgA ); |
| } |
| return count; |
| } |
| |
| static int output_message( int msg ) |
| { |
| static const WCHAR fmtW[] = {'%','s',0}; |
| WCHAR buffer[8192]; |
| |
| LoadStringW( GetModuleHandleW(NULL), msg, buffer, sizeof(buffer)/sizeof(WCHAR) ); |
| return output_string( fmtW, buffer ); |
| } |
| |
| static int query_prop( const WCHAR *class, const WCHAR *propname ) |
| { |
| static const WCHAR select_allW[] = {'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',0}; |
| static const WCHAR cimv2W[] = {'R','O','O','T','\\','C','I','M','V','2',0}; |
| static const WCHAR wqlW[] = {'W','Q','L',0}; |
| static const WCHAR newlineW[] = {'\n',0}; |
| static const WCHAR fmtW[] = {'%','s','\n',0}; |
| HRESULT hr; |
| IWbemLocator *locator = NULL; |
| IWbemServices *services = NULL; |
| IEnumWbemClassObject *result = NULL; |
| LONG flags = WBEM_FLAG_RETURN_IMMEDIATELY | WBEM_FLAG_FORWARD_ONLY; |
| BSTR path = NULL, wql = NULL, query = NULL; |
| WCHAR *prop = NULL; |
| BOOL first = TRUE; |
| int len, ret = -1; |
| |
| WINE_TRACE("%s, %s\n", debugstr_w(class), debugstr_w(propname)); |
| |
| CoInitialize( NULL ); |
| CoInitializeSecurity( NULL, -1, NULL, NULL, RPC_C_AUTHN_LEVEL_DEFAULT, |
| RPC_C_IMP_LEVEL_IMPERSONATE, NULL, EOAC_NONE, NULL ); |
| |
| hr = CoCreateInstance( &CLSID_WbemLocator, NULL, CLSCTX_INPROC_SERVER, &IID_IWbemLocator, |
| (void **)&locator ); |
| if (hr != S_OK) goto done; |
| |
| if (!(path = SysAllocString( cimv2W ))) goto done; |
| hr = IWbemLocator_ConnectServer( locator, path, NULL, NULL, NULL, 0, NULL, NULL, &services ); |
| if (hr != S_OK) goto done; |
| |
| len = strlenW( class ) + sizeof(select_allW) / sizeof(select_allW[0]); |
| if (!(query = SysAllocStringLen( NULL, len ))) goto done; |
| strcpyW( query, select_allW ); |
| strcatW( query, class ); |
| |
| if (!(wql = SysAllocString( wqlW ))) goto done; |
| hr = IWbemServices_ExecQuery( services, wql, query, flags, NULL, &result ); |
| if (hr != S_OK) goto done; |
| |
| for (;;) |
| { |
| IWbemClassObject *obj; |
| ULONG count; |
| VARIANT v; |
| |
| IEnumWbemClassObject_Next( result, WBEM_INFINITE, 1, &obj, &count ); |
| if (!count) break; |
| |
| if (first) |
| { |
| if (!(prop = find_prop( obj, propname ))) |
| { |
| output_message( STRING_INVALID_QUERY ); |
| goto done; |
| } |
| output_string( fmtW, prop ); |
| first = FALSE; |
| } |
| if (IWbemClassObject_Get( obj, prop, 0, &v, NULL, NULL ) == WBEM_S_NO_ERROR) |
| { |
| VariantChangeType( &v, &v, 0, VT_BSTR ); |
| output_string( fmtW, V_BSTR( &v ) ); |
| VariantClear( &v ); |
| } |
| IWbemClassObject_Release( obj ); |
| } |
| output_string( newlineW ); |
| ret = 0; |
| |
| done: |
| if (result) IEnumWbemClassObject_Release( result ); |
| if (services) IWbemServices_Release( services ); |
| if (locator) IWbemLocator_Release( locator ); |
| SysFreeString( path ); |
| SysFreeString( query ); |
| SysFreeString( wql ); |
| HeapFree( GetProcessHeap(), 0, prop ); |
| CoUninitialize(); |
| return ret; |
| } |
| |
| int wmain(int argc, WCHAR *argv[]) |
| { |
| static const WCHAR getW[] = {'g','e','t',0}; |
| static const WCHAR quitW[] = {'q','u','i','t',0}; |
| static const WCHAR exitW[] = {'e','x','i','t',0}; |
| static const WCHAR pathW[] = {'p','a','t','h',0}; |
| static const WCHAR classW[] = {'c','l','a','s','s',0}; |
| static const WCHAR contextW[] = {'c','o','n','t','e','x','t',0}; |
| const WCHAR *class, *value; |
| int i; |
| |
| for (i = 1; i < argc && argv[i][0] == '/'; i++) |
| WINE_FIXME( "command line switch %s not supported\n", debugstr_w(argv[i]) ); |
| |
| if (i >= argc) |
| goto not_supported; |
| |
| if (!strcmpiW( argv[i], quitW ) || |
| !strcmpiW( argv[i], exitW )) |
| { |
| return 0; |
| } |
| |
| if (!strcmpiW( argv[i], classW) || |
| !strcmpiW( argv[i], contextW )) |
| { |
| WINE_FIXME( "command %s not supported\n", debugstr_w(argv[i]) ); |
| goto not_supported; |
| } |
| |
| if (!strcmpiW( argv[i], pathW )) |
| { |
| if (++i >= argc) |
| { |
| output_message( STRING_INVALID_PATH ); |
| return 1; |
| } |
| class = argv[i]; |
| } |
| else |
| { |
| class = find_class( argv[i] ); |
| if (!class) |
| { |
| output_message( STRING_ALIAS_NOT_FOUND ); |
| return 1; |
| } |
| } |
| |
| if (++i >= argc) |
| goto not_supported; |
| |
| if (!strcmpiW( argv[i], getW )) |
| { |
| if (++i >= argc) |
| goto not_supported; |
| value = argv[i]; |
| return query_prop( class, value ); |
| } |
| |
| not_supported: |
| output_message( STRING_CMDLINE_NOT_SUPPORTED ); |
| return 1; |
| } |