2 * Copyright 2012 Hans Leidekker for CodeWeavers
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
28 #include "wine/debug.h"
29 #include "wbemprox_private.h"
31 WINE_DEFAULT_DEBUG_CHANNEL(wbemprox);
33 static HRESULT get_column_index( const struct table *table, const WCHAR *name, UINT *column )
36 for (i = 0; i < table->num_cols; i++)
38 if (!strcmpiW( table->columns[i].name, name ))
44 return WBEM_E_INVALID_QUERY;
47 static UINT get_column_size( const struct table *table, UINT column )
49 if (table->columns[column].type & CIM_FLAG_ARRAY) return sizeof(void *);
51 switch (table->columns[column].type & COL_TYPE_MASK)
64 return sizeof(WCHAR *);
66 ERR("unkown column type %u\n", table->columns[column].type & COL_TYPE_MASK);
72 static UINT get_column_offset( const struct table *table, UINT column )
75 for (i = 0; i < column; i++) offset += get_column_size( table, i );
79 static UINT get_row_size( const struct table *table )
81 return get_column_offset( table, table->num_cols - 1 ) + get_column_size( table, table->num_cols - 1 );
84 static HRESULT get_value( const struct table *table, UINT row, UINT column, LONGLONG *val )
86 UINT col_offset, row_size;
89 col_offset = get_column_offset( table, column );
90 row_size = get_row_size( table );
91 ptr = table->data + row * row_size + col_offset;
93 if (table->columns[column].type & CIM_FLAG_ARRAY)
95 *val = (LONGLONG)(INT_PTR)*(const void **)ptr;
98 switch (table->columns[column].type & COL_TYPE_MASK)
102 *val = (LONGLONG)(INT_PTR)*(const WCHAR **)ptr;
105 *val = *(const INT16 *)ptr;
108 *val = *(const UINT16 *)ptr;
111 *val = *(const INT32 *)ptr;
114 *val = *(const UINT32 *)ptr;
117 *val = *(const INT64 *)ptr;
120 *val = *(const UINT64 *)ptr;
123 ERR("invalid column type %u\n", table->columns[column].type & COL_TYPE_MASK);
130 static BSTR get_value_bstr( const struct table *table, UINT row, UINT column )
132 static const WCHAR fmt_signedW[] = {'%','d',0};
133 static const WCHAR fmt_unsignedW[] = {'%','u',0};
134 static const WCHAR fmt_signed64W[] = {'%','I','6','4','d',0};
135 static const WCHAR fmt_unsigned64W[] = {'%','I','6','4','u',0};
136 static const WCHAR fmt_strW[] = {'\"','%','s','\"',0};
142 if (table->columns[column].type & CIM_FLAG_ARRAY)
144 FIXME("array to string conversion not handled\n");
147 if (get_value( table, row, column, &val ) != S_OK) return NULL;
149 switch (table->columns[column].type & COL_TYPE_MASK)
153 len = strlenW( (const WCHAR *)(INT_PTR)val ) + 2;
154 if (!(ret = SysAllocStringLen( NULL, len ))) return NULL;
155 sprintfW( ret, fmt_strW, (const WCHAR *)(INT_PTR)val );
160 sprintfW( number, fmt_signedW, val );
161 return SysAllocString( number );
165 sprintfW( number, fmt_unsignedW, val );
166 return SysAllocString( number );
169 wsprintfW( number, fmt_signed64W, val );
170 return SysAllocString( number );
173 wsprintfW( number, fmt_unsigned64W, val );
174 return SysAllocString( number );
177 FIXME("unhandled column type %u\n", table->columns[column].type & COL_TYPE_MASK);
183 HRESULT create_view( const struct property *proplist, const WCHAR *class,
184 const struct expr *cond, struct view **ret )
186 struct view *view = heap_alloc( sizeof(struct view) );
188 if (!view) return E_OUTOFMEMORY;
189 view->proplist = proplist;
190 view->table = get_table( class );
198 static void clear_table( struct table *table )
202 if (!table->fill || !table->data) return;
204 for (i = 0; i < table->num_rows; i++)
206 for (j = 0; j < table->num_cols; j++)
208 if (!(table->columns[j].type & COL_FLAG_DYNAMIC)) continue;
210 type = table->columns[j].type & COL_TYPE_MASK;
211 if (type == CIM_STRING || type == CIM_DATETIME || (type & CIM_FLAG_ARRAY))
214 if (get_value( table, i, j, (LONGLONG *)&ptr ) == S_OK) heap_free( ptr );
218 heap_free( table->data );
222 void destroy_view( struct view *view )
224 if (view->table) clear_table( view->table );
225 heap_free( view->result );
229 static BOOL eval_like( LONGLONG lval, LONGLONG rval )
231 const WCHAR *p = (const WCHAR *)(INT_PTR)lval, *q = (const WCHAR *)(INT_PTR)rval;
237 while (*q == '%') q++;
238 if (!*q) return TRUE;
239 while (*p && toupperW( p[1] ) != toupperW( q[1] )) p++;
240 if (!*p) return TRUE;
242 if (toupperW( *p++ ) != toupperW( *q++ )) return FALSE;
247 static HRESULT eval_cond( const struct table *, UINT, const struct expr *, LONGLONG * );
249 static BOOL eval_binary( const struct table *table, UINT row, const struct complex_expr *expr,
255 lret = eval_cond( table, row, expr->left, &lval );
256 rret = eval_cond( table, row, expr->right, &rval );
257 if (lret != S_OK || rret != S_OK) return WBEM_E_INVALID_QUERY;
262 *val = (lval == rval);
265 *val = (lval && rval);
268 *val = (lval || rval);
271 *val = (lval > rval);
274 *val = (lval < rval);
277 *val = (lval <= rval);
280 *val = (lval >= rval);
283 *val = (lval != rval);
286 *val = eval_like( lval, rval );
289 ERR("unknown operator %u\n", expr->op);
290 return WBEM_E_INVALID_QUERY;
295 static HRESULT eval_unary( const struct table *table, UINT row, const struct complex_expr *expr,
303 hr = get_column_index( table, expr->left->u.propval->name, &column );
307 hr = get_value( table, row, column, &lval );
320 ERR("unknown operator %u\n", expr->op);
321 return WBEM_E_INVALID_QUERY;
326 static HRESULT eval_propval( const struct table *table, UINT row, const struct property *propval,
333 hr = get_column_index( table, propval->name, &column );
337 return get_value( table, row, column, val );
340 static HRESULT eval_cond( const struct table *table, UINT row, const struct expr *cond,
351 return eval_binary( table, row, &cond->u.expr, val );
353 return eval_unary( table, row, &cond->u.expr, val );
355 return eval_propval( table, row, cond->u.propval, val );
357 *val = (LONGLONG)(INT_PTR)cond->u.sval;
364 ERR("invalid expression type\n");
367 return WBEM_E_INVALID_QUERY;
370 static HRESULT execute_view( struct view *view )
374 if (!view->table || !view->table->num_rows) return S_OK;
376 len = min( view->table->num_rows, 16 );
377 if (!(view->result = heap_alloc( len * sizeof(UINT) ))) return E_OUTOFMEMORY;
379 for (i = 0; i < view->table->num_rows; i++)
388 if (!(tmp = heap_realloc( view->result, len * sizeof(UINT) ))) return E_OUTOFMEMORY;
391 if ((hr = eval_cond( view->table, i, view->cond, &val )) != S_OK) return hr;
392 if (val) view->result[j++] = i;
398 static struct query *create_query(void)
402 if (!(query = heap_alloc( sizeof(*query) ))) return NULL;
403 list_init( &query->mem );
408 static void free_query( struct query *query )
410 struct list *mem, *next;
412 destroy_view( query->view );
413 LIST_FOR_EACH_SAFE( mem, next, &query->mem )
420 void addref_query( struct query *query )
422 InterlockedIncrement( &query->refs );
425 void release_query( struct query *query )
427 if (!InterlockedDecrement( &query->refs )) free_query( query );
430 HRESULT exec_query( const WCHAR *str, IEnumWbemClassObject **result )
436 if (!(query = create_query())) return E_OUTOFMEMORY;
437 hr = parse_query( str, &query->view, &query->mem );
438 if (hr != S_OK) goto done;
439 hr = execute_view( query->view );
440 if (hr != S_OK) goto done;
441 hr = EnumWbemClassObject_create( NULL, query, (void **)result );
444 release_query( query );
448 static BOOL is_selected_prop( const struct view *view, const WCHAR *name )
450 const struct property *prop = view->proplist;
452 if (!prop) return TRUE;
455 if (!strcmpiW( prop->name, name )) return TRUE;
461 static BOOL is_system_prop( const WCHAR *name )
463 return (name[0] == '_' && name[1] == '_');
466 static BSTR build_servername( const struct view *view )
468 WCHAR server[MAX_COMPUTERNAME_LENGTH + 1], *p;
469 DWORD len = sizeof(server)/sizeof(server[0]);
471 if (view->proplist) return NULL;
473 if (!(GetComputerNameW( server, &len ))) return NULL;
474 for (p = server; *p; p++) *p = toupperW( *p );
475 return SysAllocString( server );
478 static BSTR build_classname( const struct view *view )
480 return SysAllocString( view->table->name );
483 static BSTR build_namespace( const struct view *view )
485 static const WCHAR cimv2W[] = {'R','O','O','T','\\','C','I','M','V','2',0};
487 if (view->proplist) return NULL;
488 return SysAllocString( cimv2W );
491 static BSTR build_proplist( const struct view *view, UINT index, UINT count, UINT *len )
493 static const WCHAR fmtW[] = {'%','s','=','%','s',0};
494 UINT i, j, offset, row = view->result[index];
495 BSTR *values, ret = NULL;
497 if (!(values = heap_alloc( count * sizeof(BSTR) ))) return NULL;
500 for (i = 0; i < view->table->num_cols; i++)
502 if (view->table->columns[i].type & COL_FLAG_KEY)
504 const WCHAR *name = view->table->columns[i].name;
506 values[j] = get_value_bstr( view->table, row, i );
507 *len += strlenW( fmtW ) + strlenW( name ) + strlenW( values[j] );
511 if ((ret = SysAllocStringLen( NULL, *len )))
514 for (i = 0; i < view->table->num_cols; i++)
516 if (view->table->columns[i].type & COL_FLAG_KEY)
518 const WCHAR *name = view->table->columns[i].name;
520 offset += sprintfW( ret + offset, fmtW, name, values[j] );
521 if (j < count - 1) ret[offset++] = ',';
526 for (i = 0; i < count; i++) SysFreeString( values[i] );
531 static UINT count_key_columns( const struct view *view )
533 UINT i, num_keys = 0;
535 for (i = 0; i < view->table->num_cols; i++)
537 if (view->table->columns[i].type & COL_FLAG_KEY) num_keys++;
542 static BSTR build_relpath( const struct view *view, UINT index, const WCHAR *name )
544 static const WCHAR fmtW[] = {'%','s','.','%','s',0};
545 BSTR class, proplist, ret = NULL;
548 if (view->proplist) return NULL;
550 if (!(class = build_classname( view ))) return NULL;
551 if (!(num_keys = count_key_columns( view ))) return class;
552 if (!(proplist = build_proplist( view, index, num_keys, &len ))) goto done;
554 len += strlenW( fmtW ) + SysStringLen( class );
555 if (!(ret = SysAllocStringLen( NULL, len ))) goto done;
556 sprintfW( ret, fmtW, class, proplist );
559 SysFreeString( class );
560 SysFreeString( proplist );
564 static BSTR build_path( const struct view *view, UINT index, const WCHAR *name )
566 static const WCHAR fmtW[] = {'\\','\\','%','s','\\','%','s',':','%','s',0};
567 BSTR server, namespace = NULL, relpath = NULL, ret = NULL;
570 if (view->proplist) return NULL;
572 if (!(server = build_servername( view ))) return NULL;
573 if (!(namespace = build_namespace( view ))) goto done;
574 if (!(relpath = build_relpath( view, index, name ))) goto done;
576 len = strlenW( fmtW ) + SysStringLen( server ) + SysStringLen( namespace ) + SysStringLen( relpath );
577 if (!(ret = SysAllocStringLen( NULL, len ))) goto done;
578 sprintfW( ret, fmtW, server, namespace, relpath );
581 SysFreeString( server );
582 SysFreeString( namespace );
583 SysFreeString( relpath );
587 static UINT count_selected_props( const struct view *view )
589 const struct property *prop = view->proplist;
592 if (!prop) return view->table->num_cols;
595 while ((prop = prop->next)) count++;
599 static HRESULT get_system_propval( const struct view *view, UINT index, const WCHAR *name,
600 VARIANT *ret, CIMTYPE *type, LONG *flavor )
602 static const WCHAR classW[] = {'_','_','C','L','A','S','S',0};
603 static const WCHAR genusW[] = {'_','_','G','E','N','U','S',0};
604 static const WCHAR pathW[] = {'_','_','P','A','T','H',0};
605 static const WCHAR namespaceW[] = {'_','_','N','A','M','E','S','P','A','C','E',0};
606 static const WCHAR propcountW[] = {'_','_','P','R','O','P','E','R','T','Y','_','C','O','U','N','T',0};
607 static const WCHAR relpathW[] = {'_','_','R','E','L','P','A','T','H',0};
608 static const WCHAR serverW[] = {'_','_','S','E','R','V','E','R',0};
610 if (flavor) *flavor = WBEM_FLAVOR_ORIGIN_SYSTEM;
612 if (!strcmpiW( name, classW ))
614 V_VT( ret ) = VT_BSTR;
615 V_BSTR( ret ) = build_classname( view );
616 if (type) *type = CIM_STRING;
619 if (!strcmpiW( name, genusW ))
622 V_I4( ret ) = WBEM_GENUS_INSTANCE; /* FIXME */
623 if (type) *type = CIM_SINT32;
626 else if (!strcmpiW( name, namespaceW ))
628 V_VT( ret ) = VT_BSTR;
629 V_BSTR( ret ) = build_namespace( view );
630 if (type) *type = CIM_STRING;
633 else if (!strcmpiW( name, pathW ))
635 V_VT( ret ) = VT_BSTR;
636 V_BSTR( ret ) = build_path( view, index, name );
637 if (type) *type = CIM_STRING;
640 if (!strcmpiW( name, propcountW ))
643 V_I4( ret ) = count_selected_props( view );
644 if (type) *type = CIM_SINT32;
647 else if (!strcmpiW( name, relpathW ))
649 V_VT( ret ) = VT_BSTR;
650 V_BSTR( ret ) = build_relpath( view, index, name );
651 if (type) *type = CIM_STRING;
654 else if (!strcmpiW( name, serverW ))
656 V_VT( ret ) = VT_BSTR;
657 V_BSTR( ret ) = build_servername( view );
658 if (type) *type = CIM_STRING;
661 FIXME("system property %s not implemented\n", debugstr_w(name));
662 return WBEM_E_NOT_FOUND;
665 HRESULT get_propval( const struct view *view, UINT index, const WCHAR *name, VARIANT *ret,
666 CIMTYPE *type, LONG *flavor )
669 UINT column, row = view->result[index];
672 if (is_system_prop( name )) return get_system_propval( view, index, name, ret, type, flavor );
673 if (!is_selected_prop( view, name )) return WBEM_E_NOT_FOUND;
675 hr = get_column_index( view->table, name, &column );
676 if (hr != S_OK) return WBEM_E_NOT_FOUND;
678 hr = get_value( view->table, row, column, &val );
679 if (hr != S_OK) return hr;
681 switch (view->table->columns[column].type & COL_TYPE_MASK)
685 V_VT( ret ) = VT_BSTR;
686 V_BSTR( ret ) = SysAllocString( (const WCHAR *)(INT_PTR)val );
693 V_VT( ret ) = VT_UI2;
701 V_VT( ret ) = VT_UI4;
709 V_VT( ret ) = VT_UI8;
713 ERR("unhandled column type %u\n", view->table->columns[column].type);
714 return WBEM_E_FAILED;
716 if (type) *type = view->table->columns[column].type & COL_TYPE_MASK;
717 if (flavor) *flavor = 0;
721 HRESULT get_properties( const struct view *view, SAFEARRAY **props )
727 if (!(sa = SafeArrayCreateVector( VT_BSTR, 0, view->table->num_cols ))) return E_OUTOFMEMORY;
729 for (i = 0; i < view->table->num_cols; i++)
731 str = SysAllocString( view->table->columns[i].name );
732 if (!str || SafeArrayPutElement( sa, &i, str ) != S_OK)
734 SysFreeString( str );
735 SafeArrayDestroy( sa );
736 return E_OUTOFMEMORY;