r17363: Some C++ warnings
[samba.git] / source / libmsrpc / libmsrpc_internal.c
1 /* 
2  *  Unix SMB/CIFS implementation.
3  *  MS-RPC client internal functions
4  *  Copyright (C) Chris Nicholls              2005.
5  *  
6  *  This program is free software; you can redistribute it and/or modify
7  *  it under the terms of the GNU General Public License as published by
8  *  the Free Software Foundation; either version 2 of the License, or
9  *  (at your option) any later version.
10  *  
11  *  This program is distributed in the hope that it will be useful,
12  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
13  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  *  GNU General Public License for more details.
15  *  
16  *  You should have received a copy of the GNU General Public License
17  *  along with this program; if not, write to the Free Software
18  *  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
19  */
20
21
22 #include "libmsrpc.h"
23 #include "libmsrpc_internal.h"
24
25 /*used to get a struct rpc_pipe_client* to be passed into rpccli* calls*/
26 struct rpc_pipe_client *cac_GetPipe(CacServerHandle *hnd, int pi_idx)
27 {
28         SMBCSRV *srv = NULL;
29         struct rpc_pipe_client *pipe_hnd = NULL;
30
31         if(!hnd) {
32                 return NULL;
33         }
34
35         if(hnd->_internal.pipes[pi_idx] == False) {
36                 hnd->status = NT_STATUS_INVALID_HANDLE;
37                 return NULL;
38         }
39
40         srv = cac_GetServer(hnd);
41         if(!srv) {
42                 hnd->status = NT_STATUS_INVALID_CONNECTION;
43                 return NULL;
44         }
45
46         pipe_hnd = srv->cli->pipe_list;
47
48         while(pipe_hnd != NULL && pipe_hnd->pipe_idx != pi_idx) {
49                 pipe_hnd = pipe_hnd->next;
50         }
51
52         return pipe_hnd;
53 }
54
55 /*takes a string like HKEY_LOCAL_MACHINE\HARDWARE\ACPI and returns the reg_type code and then a pointer to the start of the path (HARDWARE)*/
56 int cac_ParseRegPath(char *path, uint32 *reg_type, char **key_name) {
57
58    if(!path)
59       return CAC_FAILURE;
60
61    if(strncmp(path, "HKLM", 4) == 0) {
62       *reg_type = HKEY_LOCAL_MACHINE;
63       *key_name = (path[4] == '\\') ? path + 5 : NULL;
64    }
65    else if(strncmp(path, "HKEY_LOCAL_MACHINE", 18) == 0) {
66       *reg_type = HKEY_LOCAL_MACHINE;
67       *key_name = (path[18] == '\\') ? path + 19 : NULL;
68    }
69    else if(strncmp(path, "HKCR", 4) == 0) {
70       *reg_type = HKEY_CLASSES_ROOT;
71       *key_name = (path[4] == '\\') ? path + 5 : NULL;
72    }
73    else if(strncmp(path, "HKEY_CLASSES_ROOT", 17) == 0) {
74       *reg_type = HKEY_CLASSES_ROOT;
75       *key_name = (path[17] == '\\') ? path + 18 : NULL;
76    }
77    else if(strncmp(path, "HKU", 3) == 0) {
78       *reg_type = HKEY_USERS;
79       *key_name = (path[3] == '\\') ? path + 4 : NULL;
80    }
81    else if(strncmp(path, "HKEY_USERS", 10) == 0) {
82       *reg_type = HKEY_USERS;
83       *key_name = (path[10] == '\\') ? path + 11 : NULL;
84    }
85    else if(strncmp(path, "HKPD", 4) == 0) {
86       *reg_type = HKEY_PERFORMANCE_DATA;
87       *key_name = (path[4] == '\\') ? path + 5 : NULL;
88    }
89    else if(strncmp(path, "HKEY_PERFORMANCE_DATA", 21) == 0) {
90       *reg_type = HKEY_PERFORMANCE_DATA;
91       *key_name = (path[21] == '\\') ? path + 22 : NULL;
92    }
93    else {
94       return CAC_FAILURE;
95    }
96
97    return CAC_SUCCESS;
98 }
99
100
101
102 RPC_DATA_BLOB *cac_MakeRpcDataBlob(TALLOC_CTX *mem_ctx, uint32 data_type, REG_VALUE_DATA data) {
103    RPC_DATA_BLOB *blob = NULL;
104    int i;
105    uint32 size = 0;
106    uint8 *multi = NULL;
107    uint32 multi_idx = 0;
108
109    blob = talloc(mem_ctx, RPC_DATA_BLOB);
110
111    if(!blob) {
112       errno = ENOMEM;
113       return NULL;
114    }
115
116    switch(data_type) {
117       case REG_SZ:
118          init_rpc_blob_str(blob, data.reg_sz, strlen(data.reg_sz ) + 1);
119          break;
120
121       case REG_EXPAND_SZ:
122          init_rpc_blob_str(blob, data.reg_expand_sz, strlen(data.reg_sz) + 1);
123          break;
124
125       case REG_BINARY:
126          init_rpc_blob_bytes(blob, data.reg_binary.data, data.reg_binary.data_length);
127          break;
128
129       case REG_DWORD:
130          init_rpc_blob_uint32(blob, data.reg_dword);
131          break;
132          
133       case REG_DWORD_BE:
134          init_rpc_blob_uint32(blob, data.reg_dword_be);
135          break;
136          
137       case REG_MULTI_SZ:
138          /*need to find the size*/
139          for(i = 0; i < data.reg_multi_sz.num_strings; i++) {
140             size += strlen(data.reg_multi_sz.strings[i]) + 1;
141          }
142
143          /**need a whole bunch of unicode strings in a row (seperated by null characters), with an extra null-character on the end*/
144
145          multi = TALLOC_ZERO_ARRAY(mem_ctx, uint8, (size + 1)*2); /*size +1 for the extra null character*/
146          if(!multi) {
147             errno = ENOMEM;
148             break;
149          }
150          
151          /*do it using rpcstr_push()*/
152          multi_idx = 0;
153          for(i = 0; i < data.reg_multi_sz.num_strings; i++) {
154             size_t len = strlen(data.reg_multi_sz.strings[i]) + 1;
155
156             rpcstr_push((multi + multi_idx), data.reg_multi_sz.strings[i], len * 2, STR_TERMINATE);
157
158             /* x2 becuase it is a uint8 buffer*/
159             multi_idx += len * 2;
160          }
161             
162          /*now initialize the buffer as binary data*/
163          init_rpc_blob_bytes(blob, multi, (size + 1)*2);
164
165          break;
166
167       default:
168          TALLOC_FREE(blob);
169          blob = NULL;
170          return NULL;
171    }
172
173    if(!(blob->buffer)) {
174       TALLOC_FREE(blob);
175       return NULL;
176    }
177
178    return blob;
179 }
180
181 /*turns a string in a uint16 array to a char array*/
182 char *cac_unistr_to_str(TALLOC_CTX *mem_ctx, uint16 *src, int num_bytes) {
183    char *buf;
184    
185    int i = 0;
186
187    uint32 str_len = 0;
188    
189    /*don't allocate more space than we need*/
190    while( (str_len) < num_bytes/2 && src[str_len] != 0x0000)
191       str_len++;
192
193    /*need room for a '\0'*/
194    str_len++;
195
196    buf = talloc_array(mem_ctx, char, str_len);
197    if(!buf) {
198       return NULL;
199    }
200
201    for(i = 0; i < num_bytes/2; i++) {
202       buf[i] = ((char *)src)[2*i];
203    }
204
205    buf[str_len - 1] = '\0';
206
207    return buf;
208 }
209
210 REG_VALUE_DATA *cac_MakeRegValueData(TALLOC_CTX *mem_ctx, uint32 data_type, REGVAL_BUFFER buf) {
211    REG_VALUE_DATA *data;
212
213    uint32 i;
214
215    /*all of the following used for MULTI_SZ data*/
216    uint32 size       = 0;
217    uint32 len        = 0;
218    uint32 multi_idx  = 0;
219    uint32 num_strings= 0;
220    char **strings    = NULL;
221
222    data = talloc(mem_ctx, REG_VALUE_DATA);
223    if(!data) {
224       errno = ENOMEM;
225       return NULL;
226    }
227
228    switch (data_type) {
229       case REG_SZ:
230          data->reg_sz = cac_unistr_to_str(mem_ctx, buf.buffer, buf.buf_len);
231          if(!data->reg_sz) {
232             TALLOC_FREE(data);
233             errno = ENOMEM;
234             data = NULL;
235          }
236             
237          break;
238
239       case REG_EXPAND_SZ:
240          data->reg_expand_sz = cac_unistr_to_str(mem_ctx, buf.buffer, buf.buf_len);
241
242          if(!data->reg_expand_sz) {
243             TALLOC_FREE(data);
244             errno = ENOMEM;
245             data = NULL;
246          }
247             
248          break;
249
250       case REG_BINARY:
251          size = buf.buf_len;
252
253          data->reg_binary.data_length = size;
254
255          data->reg_binary.data = (uint8 *)talloc_memdup(mem_ctx, buf.buffer,
256                                                         size);
257          if(!data->reg_binary.data) {
258             TALLOC_FREE(data);
259             errno = ENOMEM;
260             data = NULL;
261          }
262          break;
263
264       case REG_DWORD:
265          data->reg_dword = *((uint32 *)buf.buffer);
266          break;
267
268       case REG_DWORD_BE:
269          data->reg_dword_be = *((uint32 *)buf.buffer);
270          break;
271
272       case REG_MULTI_SZ:
273          size = buf.buf_len;
274
275          /*find out how many strings there are. size is # of bytes and we want to work uint16*/
276          for(i = 0; i < (size/2 - 1); i++) {
277             if(buf.buffer[i] == 0x0000)
278                num_strings++;
279
280             /*buffer is suppsed to be terminated with \0\0, but it might not be*/
281             if(buf.buffer[i] == 0x0000 && buf.buffer[i + 1] == 0x0000)
282                break;
283          }
284          
285          strings = talloc_array(mem_ctx, char *, num_strings);
286          if(!strings) {
287             errno = ENOMEM;
288             TALLOC_FREE(data);
289             break;
290          }
291
292          if(num_strings == 0) /*then our work here is done*/
293             break;
294
295          for(i = 0; i < num_strings; i++) {
296             /*find out how many characters are in this string*/
297             len = 0;
298             /*make sure we don't go past the end of the buffer and keep looping until we have a uni \0*/
299             while( multi_idx + len < size/2 && buf.buffer[multi_idx + len] != 0x0000)
300                len++;
301
302             /*stay aware of the \0\0*/
303             len++;
304
305             strings[i] = TALLOC_ZERO_ARRAY(mem_ctx, char, len);
306
307             /*pull out the unicode string*/
308             rpcstr_pull(strings[i], (buf.buffer + multi_idx) , len, -1, STR_TERMINATE);
309
310             /*keep track of where we are in the bigger array*/
311             multi_idx += len;
312          }
313
314          data->reg_multi_sz.num_strings = num_strings;
315          data->reg_multi_sz.strings     = strings;
316
317          break;
318
319       default:
320          TALLOC_FREE(data);
321          data = NULL;
322    }
323
324    return data;
325 }
326
327 SAM_USERINFO_CTR *cac_MakeUserInfoCtr(TALLOC_CTX *mem_ctx, CacUserInfo *info) {
328    SAM_USERINFO_CTR *ctr = NULL;
329
330    /*the flags we are 'setting'- include/passdb.h*/
331    uint32 flags = ACCT_USERNAME | ACCT_FULL_NAME | ACCT_PRIMARY_GID | ACCT_DESCRIPTION | ACCT_COMMENT |
332                      ACCT_HOME_DIR | ACCT_HOME_DRIVE | ACCT_LOGON_SCRIPT | ACCT_PROFILE | ACCT_WORKSTATIONS |
333                       ACCT_FLAGS;
334
335    NTTIME logon_time;
336    NTTIME logoff_time;
337    NTTIME kickoff_time;
338    NTTIME pass_last_set_time;
339    NTTIME pass_can_change_time;
340    NTTIME pass_must_change_time;
341
342    UNISTR2 user_name;
343    UNISTR2 full_name;
344    UNISTR2 home_dir;
345    UNISTR2 dir_drive;
346    UNISTR2 log_scr;
347    UNISTR2 prof_path;
348    UNISTR2 desc;
349    UNISTR2 wkstas;
350    UNISTR2 mung_dial;
351    UNISTR2 unk;
352
353    ctr = talloc(mem_ctx, SAM_USERINFO_CTR);
354    if(!ctr)
355       return NULL;
356
357    ZERO_STRUCTP(ctr->info.id23);
358
359    ctr->info.id21 = talloc(mem_ctx, SAM_USER_INFO_21);
360    if(!ctr->info.id21)
361       return NULL;
362
363    ctr->switch_value = 21;
364
365    ZERO_STRUCTP(ctr->info.id21);
366
367    unix_to_nt_time(&logon_time, info->logon_time);
368    unix_to_nt_time(&logoff_time, info->logoff_time);
369    unix_to_nt_time(&kickoff_time, info->kickoff_time);
370    unix_to_nt_time(&pass_last_set_time, info->pass_last_set_time);
371    unix_to_nt_time(&pass_can_change_time, info->pass_can_change_time);
372    unix_to_nt_time(&pass_must_change_time, info->pass_must_change_time);
373
374    /*initialize the strings*/
375    init_unistr2(&user_name, info->username, UNI_STR_TERMINATE);
376    init_unistr2(&full_name, info->full_name, UNI_STR_TERMINATE);
377    init_unistr2(&home_dir, info->home_dir, UNI_STR_TERMINATE);
378    init_unistr2(&dir_drive, info->home_drive, UNI_STR_TERMINATE);
379    init_unistr2(&log_scr, info->logon_script, UNI_STR_TERMINATE);
380    init_unistr2(&prof_path, info->profile_path, UNI_STR_TERMINATE);
381    init_unistr2(&desc, info->description, UNI_STR_TERMINATE);
382    init_unistr2(&wkstas, info->workstations, UNI_STR_TERMINATE);
383    init_unistr2(&unk, "\0", UNI_STR_TERMINATE);
384    init_unistr2(&mung_dial, info->dial, UNI_STR_TERMINATE);
385
386    /*manually set passmustchange*/
387    ctr->info.id21->passmustchange = (info->pass_must_change) ? 0x01 : 0x00;
388
389    init_sam_user_info21W(ctr->info.id21,
390                          &logon_time,
391                          &logoff_time,
392                          &kickoff_time,
393                          &pass_last_set_time,
394                          &pass_can_change_time,
395                          &pass_must_change_time,
396                          &user_name,
397                          &full_name,
398                          &home_dir,
399                          &dir_drive,
400                          &log_scr,
401                          &prof_path,
402                          &desc,
403                          &wkstas,
404                          &unk,
405                          &mung_dial,
406                          info->lm_password,
407                          info->nt_password,
408                          info->rid,
409                          info->group_rid,
410                          info->acb_mask,
411                          flags,
412                          168, /*logon divs*/
413                          info->logon_hours,
414                          info->bad_passwd_count,
415                          info->logon_count);
416
417    return ctr;
418    
419 }
420
421 char *talloc_unistr2_to_ascii(TALLOC_CTX *mem_ctx, UNISTR2 str) {
422    char *buf = NULL;
423
424    if(!mem_ctx)
425       return NULL;
426
427    buf = talloc_array(mem_ctx, char, (str.uni_str_len + 1));
428    if(!buf)
429       return NULL;
430
431    unistr2_to_ascii(buf, &str, str.uni_str_len + 1);
432
433    return buf;
434 }
435
436 CacUserInfo *cac_MakeUserInfo(TALLOC_CTX *mem_ctx, SAM_USERINFO_CTR *ctr) {
437    CacUserInfo *info = NULL;
438    SAM_USER_INFO_21 *id21 = NULL;
439
440    if(!ctr || ctr->switch_value != 21)
441       return NULL;
442
443    info = talloc(mem_ctx, CacUserInfo);
444    if(!info)
445       return NULL;
446
447    id21 = ctr->info.id21;
448
449    ZERO_STRUCTP(info);
450
451    info->logon_time = nt_time_to_unix(&id21->logon_time);
452    info->logoff_time = nt_time_to_unix(&id21->logoff_time);
453    info->kickoff_time = nt_time_to_unix(&id21->kickoff_time);
454    info->pass_last_set_time = nt_time_to_unix(&id21->pass_last_set_time);
455    info->pass_can_change_time = nt_time_to_unix(&id21->pass_can_change_time);
456    info->pass_must_change_time = nt_time_to_unix(&id21->pass_must_change_time);
457
458    info->username = talloc_unistr2_to_ascii(mem_ctx, id21->uni_user_name);
459    if(!info->username)
460       return NULL;
461
462    info->full_name = talloc_unistr2_to_ascii(mem_ctx, id21->uni_full_name);
463    if(!info->full_name)
464       return NULL;
465    
466    info->home_dir = talloc_unistr2_to_ascii(mem_ctx, id21->uni_home_dir);
467    if(!info->home_dir)
468       return NULL;
469    
470    info->home_drive = talloc_unistr2_to_ascii(mem_ctx, id21->uni_dir_drive);
471    if(!info->home_drive)
472       return NULL;
473
474    info->logon_script = talloc_unistr2_to_ascii(mem_ctx, id21->uni_logon_script);
475    if(!info->logon_script)
476       return NULL;
477
478    info->profile_path = talloc_unistr2_to_ascii(mem_ctx, id21->uni_profile_path);
479    if(!info->profile_path)
480       return NULL;
481    
482    info->description = talloc_unistr2_to_ascii(mem_ctx, id21->uni_acct_desc);
483    if(!info->description)
484       return NULL;
485    
486    info->workstations = talloc_unistr2_to_ascii(mem_ctx, id21->uni_workstations);
487    if(!info->workstations)
488       return NULL;
489
490    info->dial = talloc_unistr2_to_ascii(mem_ctx, id21->uni_munged_dial);
491    if(!info->dial)
492       return NULL;
493
494    info->rid = id21->user_rid;
495    info->group_rid = id21->group_rid;
496    info->acb_mask = id21->acb_info;
497    info->bad_passwd_count = id21->bad_password_count;
498    info->logon_count = id21->logon_count;
499
500    memcpy(info->nt_password, id21->nt_pwd, 8);
501    memcpy(info->lm_password, id21->lm_pwd, 8);
502    
503    info->logon_hours = (LOGON_HRS *)talloc_memdup(mem_ctx, &(id21->logon_hrs),
504                                                   sizeof(LOGON_HRS));
505    if(!info->logon_hours)
506       return NULL;
507
508    info->pass_must_change = (id21->passmustchange) ? True : False;
509
510    return info;
511 }
512
513 CacGroupInfo *cac_MakeGroupInfo(TALLOC_CTX *mem_ctx, GROUP_INFO_CTR *ctr) {
514    CacGroupInfo *info = NULL;
515
516    if(!mem_ctx || !ctr || ctr->switch_value1 != 1)
517       return NULL;
518
519    info = talloc(mem_ctx, CacGroupInfo);
520    if(!info)
521       return NULL;
522
523    info->name = talloc_unistr2_to_ascii(mem_ctx, ctr->group.info1.uni_acct_name);
524    if(!info->name)
525       return NULL;
526
527    info->description = talloc_unistr2_to_ascii(mem_ctx, ctr->group.info1.uni_acct_desc);
528    if(!info->description)
529       return NULL;
530
531    info->num_members = ctr->group.info1.num_members;
532
533    return info;
534 }
535
536 GROUP_INFO_CTR *cac_MakeGroupInfoCtr(TALLOC_CTX *mem_ctx, CacGroupInfo *info) {
537    GROUP_INFO_CTR *ctr = NULL;
538
539    if(!mem_ctx || !info)
540       return NULL;
541
542    ctr = talloc(mem_ctx, GROUP_INFO_CTR);
543    if(!ctr)
544       return NULL;
545
546    ctr->switch_value1 = 1;
547
548    init_samr_group_info1(&(ctr->group.info1), info->name, info->description, info->num_members);
549
550    return ctr;
551 }
552
553 CacAliasInfo *cac_MakeAliasInfo(TALLOC_CTX *mem_ctx, ALIAS_INFO_CTR ctr) {
554    CacGroupInfo *info = NULL;
555
556    if(!mem_ctx || ctr.level != 1)
557       return NULL;
558
559    info = talloc(mem_ctx, CacAliasInfo);
560    if(!info)
561       return NULL;
562
563    info->name = talloc_unistr2_to_ascii(mem_ctx, *(ctr.alias.info1.name.string));
564    if(!info->name)
565       return NULL;
566
567    info->description = talloc_unistr2_to_ascii(mem_ctx, *(ctr.alias.info1.description.string));
568    if(!info->name)
569       return NULL;
570
571    info->num_members = ctr.alias.info1.num_member;
572
573    return info;
574 }
575
576 ALIAS_INFO_CTR *cac_MakeAliasInfoCtr(TALLOC_CTX *mem_ctx, CacAliasInfo *info) {
577    ALIAS_INFO_CTR *ctr = NULL;
578
579    if(!mem_ctx || !info)
580       return NULL;
581
582    ctr = talloc(mem_ctx, ALIAS_INFO_CTR);
583    if(!ctr)
584       return NULL;
585
586    ctr->level = 1;
587
588    init_samr_alias_info1(&(ctr->alias.info1), info->name, info->num_members, info->description);
589
590    return ctr;
591 }
592
593 CacDomainInfo *cac_MakeDomainInfo(TALLOC_CTX *mem_ctx, SAM_UNK_INFO_1 *info1, SAM_UNK_INFO_2 *info2, SAM_UNK_INFO_12 *info12) {
594    CacDomainInfo *info = NULL;
595
596    if(!mem_ctx || !info1 || !info2 || !info12)
597       return NULL;
598
599    info = talloc(mem_ctx, CacDomainInfo);
600    if(!info)
601       return NULL;
602
603    info->min_pass_length = info1->min_length_password;
604    info->pass_history    = info1->password_history;
605
606    cac_InitCacTime(&(info->expire), info1->expire);
607    cac_InitCacTime(&(info->min_pass_age), info1->min_passwordage);
608
609    info->server_role       = info2->server_role;
610    info->num_users         = info2->num_domain_usrs;
611    info->num_domain_groups = info2->num_domain_grps;
612    info->num_local_groups  = info2->num_local_grps;
613
614    /*if these have been ZERO'd out we need to know. uni_str_len will be 0*/
615    if(info2->uni_comment.uni_str_len == 0) {
616       info->comment = talloc_strdup(mem_ctx, "\0");
617    }
618    else {
619       info->comment = talloc_unistr2_to_ascii(mem_ctx, info2->uni_comment);
620    }
621
622    if(info2->uni_domain.uni_str_len == 0) {
623       info->domain_name = talloc_strdup(mem_ctx, "\0");
624    }
625    else {
626       info->domain_name = talloc_unistr2_to_ascii(mem_ctx, info2->uni_domain);
627    }
628
629    if(info2->uni_server.uni_str_len == 0) {
630       info->server_name = talloc_strdup(mem_ctx, "\0");
631    }
632    else {
633       info->server_name = talloc_unistr2_to_ascii(mem_ctx, info2->uni_server);
634    }
635
636
637    cac_InitCacTime(&(info->lockout_duration), info12->duration);
638    cac_InitCacTime(&(info->lockout_reset), info12->reset_count);
639    info->num_bad_attempts = info12->bad_attempt_lockout;
640
641    return info;
642 }
643
644 char *cac_unistr_ascii(TALLOC_CTX *mem_ctx, UNISTR src) {
645    char *buf;
646    uint32 len;
647
648    if(!mem_ctx || !src.buffer)
649       return NULL;
650
651    len = unistrlen(src.buffer) + 1;
652
653    buf = TALLOC_ZERO_ARRAY(mem_ctx, char, len);
654    if(!buf)
655       return NULL;
656
657    rpcstr_pull(buf, src.buffer, len, -1, STR_TERMINATE);
658
659    return buf;
660 }
661
662 CacService *cac_MakeServiceArray(TALLOC_CTX *mem_ctx, ENUM_SERVICES_STATUS *svc, uint32 num_services) {
663    int i;
664    CacService *services = NULL;
665
666    if(!mem_ctx || !svc)
667       return NULL;
668
669    services = TALLOC_ZERO_ARRAY(mem_ctx, CacService, num_services);
670    if(!services)
671       return NULL;
672
673    for(i = 0; i < num_services; i++) {
674       services[i].service_name = cac_unistr_ascii(mem_ctx, svc[i].servicename);
675       services[i].display_name = cac_unistr_ascii(mem_ctx, svc[i].displayname);
676
677       if(!services[i].service_name || !services[i].display_name)
678          return NULL;
679
680       services[i].status = svc[i].status;
681    }
682
683    return services;
684 }
685
686 int cac_InitCacServiceConfig(TALLOC_CTX *mem_ctx, SERVICE_CONFIG *src, CacServiceConfig *dest) {
687    if(!src || !dest)
688       return CAC_FAILURE;
689
690    dest->exe_path = talloc_unistr2_to_ascii(mem_ctx, *src->executablepath);
691    if(!dest->exe_path)
692       return CAC_FAILURE;
693
694    dest->load_order_group = talloc_unistr2_to_ascii(mem_ctx, *src->loadordergroup);
695    if(!dest->load_order_group)
696       return CAC_FAILURE;
697
698    dest->dependencies = talloc_unistr2_to_ascii(mem_ctx, *src->dependencies);
699    if(!dest->dependencies)
700       return CAC_FAILURE;
701
702    dest->start_name = talloc_unistr2_to_ascii(mem_ctx, *src->startname);
703    if(!dest->start_name)
704       return CAC_FAILURE;
705
706    dest->display_name = talloc_unistr2_to_ascii(mem_ctx, *src->displayname);
707    if(!dest->display_name)
708       return CAC_FAILURE;
709
710    dest->type = src->service_type;
711    dest->start_type = src->start_type;
712    dest->error_control = src->error_control;
713    dest->tag_id = src->tag_id;
714
715    return CAC_SUCCESS;
716 }