1264 lines
34 KiB
C
1264 lines
34 KiB
C
/*
|
|
* ATSvc RPC API
|
|
*
|
|
* Copyright 2018 Dmitry Timoshkov
|
|
*
|
|
* 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
|
|
*/
|
|
|
|
#include <stdarg.h>
|
|
|
|
#define NONAMELESSUNION
|
|
#include "windef.h"
|
|
#include "atsvc.h"
|
|
#include "mstask.h"
|
|
#include "wine/list.h"
|
|
#include "wine/debug.h"
|
|
|
|
#include "schedsvc_private.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(schedsvc);
|
|
|
|
/* lmat.h defines those, but other types in that file conflict
|
|
* with generated atsvc.h typedefs.
|
|
*/
|
|
#define JOB_ADD_CURRENT_DATE 0x08
|
|
#define JOB_NONINTERACTIVE 0x10
|
|
|
|
typedef struct
|
|
{
|
|
USHORT product_version;
|
|
USHORT file_version;
|
|
UUID uuid;
|
|
USHORT name_size_offset;
|
|
USHORT trigger_offset;
|
|
USHORT error_retry_count;
|
|
USHORT error_retry_interval;
|
|
USHORT idle_deadline;
|
|
USHORT idle_wait;
|
|
UINT priority;
|
|
UINT maximum_runtime;
|
|
UINT exit_code;
|
|
UINT status;
|
|
UINT flags;
|
|
SYSTEMTIME last_runtime;
|
|
} FIXDLEN_DATA;
|
|
|
|
struct job_t
|
|
{
|
|
struct list entry;
|
|
WCHAR *name;
|
|
WCHAR *params;
|
|
WCHAR *curdir;
|
|
AT_ENUM info;
|
|
FIXDLEN_DATA data;
|
|
USHORT instance_count;
|
|
USHORT trigger_count;
|
|
TASK_TRIGGER *trigger;
|
|
};
|
|
|
|
struct running_job_t
|
|
{
|
|
struct list entry;
|
|
UUID uuid;
|
|
HANDLE process;
|
|
DWORD pid;
|
|
};
|
|
|
|
static LONG current_jobid = 1;
|
|
|
|
static struct list at_job_list = LIST_INIT(at_job_list);
|
|
static struct list running_job_list = LIST_INIT(running_job_list);
|
|
|
|
static CRITICAL_SECTION at_job_list_section;
|
|
static CRITICAL_SECTION_DEBUG cs_debug =
|
|
{
|
|
0, 0, &at_job_list_section,
|
|
{ &cs_debug.ProcessLocksList, &cs_debug.ProcessLocksList },
|
|
0, 0, { (DWORD_PTR)(__FILE__ ": at_job_list_section") }
|
|
};
|
|
static CRITICAL_SECTION at_job_list_section = { &cs_debug, -1, 0, 0, 0, 0 };
|
|
|
|
static void filetime_add_ms(FILETIME *ft, LONGLONG ms)
|
|
{
|
|
union u_ftll
|
|
{
|
|
FILETIME ft;
|
|
LONGLONG ll;
|
|
} *ftll = (union u_ftll *)ft;
|
|
|
|
ftll->ll += ms * (ULONGLONG)10000;
|
|
}
|
|
|
|
static void filetime_add_minutes(FILETIME *ft, LONG minutes)
|
|
{
|
|
filetime_add_ms(ft, (LONGLONG)minutes * 60 * 1000);
|
|
}
|
|
|
|
static void filetime_add_hours(FILETIME *ft, LONG hours)
|
|
{
|
|
filetime_add_minutes(ft, (LONGLONG)hours * 60);
|
|
}
|
|
|
|
static void filetime_add_days(FILETIME *ft, LONG days)
|
|
{
|
|
filetime_add_hours(ft, (LONGLONG)days * 24);
|
|
}
|
|
|
|
static void filetime_add_weeks(FILETIME *ft, ULONG weeks)
|
|
{
|
|
filetime_add_days(ft, (LONGLONG)weeks * 7);
|
|
}
|
|
|
|
static void get_begin_time(const TASK_TRIGGER *trigger, FILETIME *ft)
|
|
{
|
|
SYSTEMTIME st;
|
|
|
|
st.wYear = trigger->wBeginYear;
|
|
st.wMonth = trigger->wBeginMonth;
|
|
st.wDay = trigger->wBeginDay;
|
|
st.wDayOfWeek = 0;
|
|
st.wHour = 0;
|
|
st.wMinute = 0;
|
|
st.wSecond = 0;
|
|
st.wMilliseconds = 0;
|
|
SystemTimeToFileTime(&st, ft);
|
|
}
|
|
|
|
static void get_end_time(const TASK_TRIGGER *trigger, FILETIME *ft)
|
|
{
|
|
SYSTEMTIME st;
|
|
|
|
if (!(trigger->rgFlags & TASK_TRIGGER_FLAG_HAS_END_DATE))
|
|
{
|
|
ft->dwHighDateTime = ~0u;
|
|
ft->dwLowDateTime = ~0u;
|
|
return;
|
|
}
|
|
|
|
st.wYear = trigger->wEndYear;
|
|
st.wMonth = trigger->wEndMonth;
|
|
st.wDay = trigger->wEndDay;
|
|
st.wDayOfWeek = 0;
|
|
st.wHour = 0;
|
|
st.wMinute = 0;
|
|
st.wSecond = 0;
|
|
st.wMilliseconds = 0;
|
|
SystemTimeToFileTime(&st, ft);
|
|
}
|
|
|
|
static BOOL trigger_get_next_runtime(const TASK_TRIGGER *trigger, const FILETIME *current_ft, FILETIME *rt)
|
|
{
|
|
SYSTEMTIME st, current_st;
|
|
FILETIME begin_ft, end_ft, trigger_ft;
|
|
|
|
if (trigger->rgFlags & TASK_TRIGGER_FLAG_DISABLED)
|
|
return FALSE;
|
|
|
|
FileTimeToSystemTime(current_ft, ¤t_st);
|
|
|
|
get_begin_time(trigger, &begin_ft);
|
|
get_end_time(trigger, &end_ft);
|
|
|
|
switch (trigger->TriggerType)
|
|
{
|
|
case TASK_EVENT_TRIGGER_ON_IDLE:
|
|
case TASK_EVENT_TRIGGER_AT_SYSTEMSTART:
|
|
case TASK_EVENT_TRIGGER_AT_LOGON:
|
|
return FALSE;
|
|
|
|
case TASK_TIME_TRIGGER_ONCE:
|
|
st = current_st;
|
|
st.wHour = trigger->wStartHour;
|
|
st.wMinute = trigger->wStartMinute;
|
|
st.wSecond = 0;
|
|
st.wMilliseconds = 0;
|
|
SystemTimeToFileTime(&st, &trigger_ft);
|
|
if (CompareFileTime(&begin_ft, &trigger_ft) <= 0 && CompareFileTime(&trigger_ft, &end_ft) < 0)
|
|
{
|
|
*rt = trigger_ft;
|
|
return TRUE;
|
|
}
|
|
break;
|
|
|
|
case TASK_TIME_TRIGGER_DAILY:
|
|
st = current_st;
|
|
st.wHour = trigger->wStartHour;
|
|
st.wMinute = trigger->wStartMinute;
|
|
st.wSecond = 0;
|
|
st.wMilliseconds = 0;
|
|
SystemTimeToFileTime(&st, &trigger_ft);
|
|
while (CompareFileTime(&trigger_ft, &end_ft) < 0)
|
|
{
|
|
if (CompareFileTime(&trigger_ft, &begin_ft) >= 0)
|
|
{
|
|
*rt = trigger_ft;
|
|
return TRUE;
|
|
}
|
|
|
|
filetime_add_days(&trigger_ft, trigger->Type.Daily.DaysInterval);
|
|
}
|
|
break;
|
|
|
|
case TASK_TIME_TRIGGER_WEEKLY:
|
|
if (!trigger->Type.Weekly.rgfDaysOfTheWeek)
|
|
break; /* avoid infinite loop */
|
|
|
|
st = current_st;
|
|
st.wHour = trigger->wStartHour;
|
|
st.wMinute = trigger->wStartMinute;
|
|
st.wSecond = 0;
|
|
st.wMilliseconds = 0;
|
|
SystemTimeToFileTime(&st, &trigger_ft);
|
|
while (CompareFileTime(&trigger_ft, &end_ft) < 0)
|
|
{
|
|
FileTimeToSystemTime(&trigger_ft, &st);
|
|
|
|
if (CompareFileTime(&trigger_ft, &begin_ft) >= 0)
|
|
{
|
|
if (trigger->Type.Weekly.rgfDaysOfTheWeek & (1 << st.wDayOfWeek))
|
|
{
|
|
*rt = trigger_ft;
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
if (st.wDayOfWeek == 0 && trigger->Type.Weekly.WeeksInterval > 1) /* Sunday, goto next week */
|
|
filetime_add_weeks(&trigger_ft, trigger->Type.Weekly.WeeksInterval - 1);
|
|
else /* check next weekday */
|
|
filetime_add_days(&trigger_ft, 1);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
FIXME("trigger type %u is not handled\n", trigger->TriggerType);
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static BOOL job_get_next_runtime(struct job_t *job, FILETIME *current_ft, FILETIME *next_rt)
|
|
{
|
|
FILETIME trigger_rt;
|
|
BOOL have_next_rt = FALSE;
|
|
USHORT i;
|
|
|
|
for (i = 0; i < job->trigger_count; i++)
|
|
{
|
|
if (trigger_get_next_runtime(&job->trigger[i], current_ft, &trigger_rt))
|
|
{
|
|
if (!have_next_rt || CompareFileTime(&trigger_rt, next_rt) < 0)
|
|
{
|
|
*next_rt = trigger_rt;
|
|
have_next_rt = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
return have_next_rt;
|
|
}
|
|
|
|
/* Returns next runtime in UTC */
|
|
BOOL get_next_runtime(LARGE_INTEGER *rt)
|
|
{
|
|
FILETIME current_ft, job_rt, next_job_rt;
|
|
BOOL have_next_rt = FALSE;
|
|
struct job_t *job;
|
|
|
|
GetSystemTimeAsFileTime(¤t_ft);
|
|
FileTimeToLocalFileTime(¤t_ft, ¤t_ft);
|
|
|
|
EnterCriticalSection(&at_job_list_section);
|
|
|
|
LIST_FOR_EACH_ENTRY(job, &at_job_list, struct job_t, entry)
|
|
{
|
|
if (job_get_next_runtime(job, ¤t_ft, &job_rt))
|
|
{
|
|
if (!have_next_rt || CompareFileTime(&job_rt, &next_job_rt) < 0)
|
|
{
|
|
next_job_rt = job_rt;
|
|
have_next_rt = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
LeaveCriticalSection(&at_job_list_section);
|
|
|
|
if (have_next_rt)
|
|
{
|
|
LocalFileTimeToFileTime(&next_job_rt, &next_job_rt);
|
|
rt->u.LowPart = next_job_rt.dwLowDateTime;
|
|
rt->u.HighPart = next_job_rt.dwHighDateTime;
|
|
}
|
|
|
|
return have_next_rt;
|
|
}
|
|
|
|
static BOOL job_runs_at(struct job_t *job, FILETIME *begin_ft, FILETIME *end_ft)
|
|
{
|
|
FILETIME job_ft;
|
|
|
|
if (job_get_next_runtime(job, begin_ft, &job_ft))
|
|
{
|
|
if (CompareFileTime(&job_ft, end_ft) < 0)
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static DWORD load_unicode_strings(const char *data, DWORD limit, struct job_t *job)
|
|
{
|
|
DWORD i, data_size = 0;
|
|
USHORT len;
|
|
|
|
for (i = 0; i < 5; i++)
|
|
{
|
|
if (limit < sizeof(USHORT))
|
|
{
|
|
TRACE("invalid string %u offset\n", i);
|
|
break;
|
|
}
|
|
|
|
len = *(USHORT *)data;
|
|
data += sizeof(USHORT);
|
|
data_size += sizeof(USHORT);
|
|
limit -= sizeof(USHORT);
|
|
if (limit < len * sizeof(WCHAR))
|
|
{
|
|
TRACE("invalid string %u size\n", i);
|
|
break;
|
|
}
|
|
|
|
TRACE("string %u: %s\n", i, wine_dbgstr_wn((const WCHAR *)data, len));
|
|
|
|
switch (i)
|
|
{
|
|
case 0:
|
|
job->info.Command = heap_strdupW((const WCHAR *)data);
|
|
break;
|
|
|
|
case 1:
|
|
job->params = heap_strdupW((const WCHAR *)data);
|
|
break;
|
|
|
|
case 2:
|
|
job->curdir = heap_strdupW((const WCHAR *)data);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
data += len * sizeof(WCHAR);
|
|
data_size += len * sizeof(WCHAR);
|
|
}
|
|
|
|
return data_size;
|
|
}
|
|
|
|
static BOOL load_job_data(const char *data, DWORD size, struct job_t *info)
|
|
{
|
|
const FIXDLEN_DATA *fixed;
|
|
const SYSTEMTIME *st;
|
|
DWORD unicode_strings_size, data_size, triggers_size;
|
|
USHORT i;
|
|
const USHORT *signature;
|
|
const TASK_TRIGGER *trigger;
|
|
|
|
memset(info, 0, sizeof(*info));
|
|
|
|
if (size < sizeof(*fixed))
|
|
{
|
|
TRACE("no space for FIXDLEN_DATA\n");
|
|
return FALSE;
|
|
}
|
|
|
|
fixed = (const FIXDLEN_DATA *)data;
|
|
info->data = *fixed;
|
|
|
|
TRACE("product_version %04x\n", fixed->product_version);
|
|
TRACE("file_version %04x\n", fixed->file_version);
|
|
TRACE("uuid %s\n", wine_dbgstr_guid(&fixed->uuid));
|
|
|
|
if (fixed->file_version != 0x0001)
|
|
{
|
|
TRACE("invalid file version\n");
|
|
return FALSE;
|
|
}
|
|
|
|
TRACE("name_size_offset %04x\n", fixed->name_size_offset);
|
|
TRACE("trigger_offset %04x\n", fixed->trigger_offset);
|
|
TRACE("error_retry_count %u\n", fixed->error_retry_count);
|
|
TRACE("error_retry_interval %u\n", fixed->error_retry_interval);
|
|
TRACE("idle_deadline %u\n", fixed->idle_deadline);
|
|
TRACE("idle_wait %u\n", fixed->idle_wait);
|
|
TRACE("priority %08x\n", fixed->priority);
|
|
TRACE("maximum_runtime %u\n", fixed->maximum_runtime);
|
|
TRACE("exit_code %#x\n", fixed->exit_code);
|
|
TRACE("status %08x\n", fixed->status);
|
|
TRACE("flags %08x\n", fixed->flags);
|
|
st = &fixed->last_runtime;
|
|
TRACE("last_runtime %d/%d/%d wday %d %d:%d:%d.%03d\n",
|
|
st->wDay, st->wMonth, st->wYear, st->wDayOfWeek,
|
|
st->wHour, st->wMinute, st->wSecond, st->wMilliseconds);
|
|
|
|
/* Instance Count */
|
|
if (size < sizeof(*fixed) + sizeof(USHORT))
|
|
{
|
|
TRACE("no space for instance count\n");
|
|
return FALSE;
|
|
}
|
|
|
|
info->instance_count = *(const USHORT *)(data + sizeof(*fixed));
|
|
TRACE("instance count %u\n", info->instance_count);
|
|
|
|
if (fixed->name_size_offset + sizeof(USHORT) < size)
|
|
unicode_strings_size = load_unicode_strings(data + fixed->name_size_offset, size - fixed->name_size_offset, info);
|
|
else
|
|
{
|
|
TRACE("invalid name_size_offset\n");
|
|
return FALSE;
|
|
}
|
|
TRACE("unicode strings end at %#x\n", fixed->name_size_offset + unicode_strings_size);
|
|
|
|
if (size < fixed->trigger_offset + sizeof(USHORT))
|
|
{
|
|
TRACE("no space for triggers count\n");
|
|
return FALSE;
|
|
}
|
|
info->trigger_count = *(const USHORT *)(data + fixed->trigger_offset);
|
|
TRACE("trigger_count %u\n", info->trigger_count);
|
|
triggers_size = size - fixed->trigger_offset - sizeof(USHORT);
|
|
TRACE("triggers_size %u\n", triggers_size);
|
|
|
|
data += fixed->name_size_offset + unicode_strings_size;
|
|
size -= fixed->name_size_offset + unicode_strings_size;
|
|
|
|
/* User Data */
|
|
if (size < sizeof(USHORT))
|
|
{
|
|
TRACE("no space for user data size\n");
|
|
return FALSE;
|
|
}
|
|
|
|
data_size = *(const USHORT *)data;
|
|
if (size < sizeof(USHORT) + data_size)
|
|
{
|
|
TRACE("no space for user data\n");
|
|
return FALSE;
|
|
}
|
|
TRACE("User Data size %#x\n", data_size);
|
|
|
|
size -= sizeof(USHORT) + data_size;
|
|
data += sizeof(USHORT) + data_size;
|
|
|
|
/* Reserved Data */
|
|
if (size < sizeof(USHORT))
|
|
{
|
|
TRACE("no space for reserved data size\n");
|
|
return FALSE;
|
|
}
|
|
|
|
data_size = *(const USHORT *)data;
|
|
if (size < sizeof(USHORT) + data_size)
|
|
{
|
|
TRACE("no space for reserved data\n");
|
|
return FALSE;
|
|
}
|
|
TRACE("Reserved Data size %#x\n", data_size);
|
|
|
|
size -= sizeof(USHORT) + data_size;
|
|
data += sizeof(USHORT) + data_size;
|
|
|
|
/* Trigger Data */
|
|
TRACE("trigger_offset %04x, triggers end at %04x\n", fixed->trigger_offset,
|
|
(DWORD)(fixed->trigger_offset + sizeof(USHORT) + info->trigger_count * sizeof(TASK_TRIGGER)));
|
|
|
|
info->trigger_count = *(const USHORT *)data;
|
|
TRACE("trigger_count %u\n", info->trigger_count);
|
|
trigger = (const TASK_TRIGGER *)(data + sizeof(USHORT));
|
|
|
|
if (info->trigger_count * sizeof(TASK_TRIGGER) > triggers_size)
|
|
{
|
|
TRACE("no space for triggers data\n");
|
|
return FALSE;
|
|
}
|
|
|
|
info->trigger = heap_alloc(info->trigger_count * sizeof(info->trigger[0]));
|
|
if (!info->trigger)
|
|
{
|
|
TRACE("not enough memory for trigger data\n");
|
|
return FALSE;
|
|
}
|
|
|
|
for (i = 0; i < info->trigger_count; i++)
|
|
{
|
|
TRACE("%u: cbTriggerSize = %#x\n", i, trigger[i].cbTriggerSize);
|
|
if (trigger[i].cbTriggerSize != sizeof(TASK_TRIGGER))
|
|
TRACE("invalid cbTriggerSize\n");
|
|
TRACE("Reserved1 = %#x\n", trigger[i].Reserved1);
|
|
TRACE("wBeginYear = %u\n", trigger[i].wBeginYear);
|
|
TRACE("wBeginMonth = %u\n", trigger[i].wBeginMonth);
|
|
TRACE("wBeginDay = %u\n", trigger[i].wBeginDay);
|
|
TRACE("wEndYear = %u\n", trigger[i].wEndYear);
|
|
TRACE("wEndMonth = %u\n", trigger[i].wEndMonth);
|
|
TRACE("wEndDay = %u\n", trigger[i].wEndDay);
|
|
TRACE("wStartHour = %u\n", trigger[i].wStartHour);
|
|
TRACE("wStartMinute = %u\n", trigger[i].wStartMinute);
|
|
TRACE("MinutesDuration = %u\n", trigger[i].MinutesDuration);
|
|
TRACE("MinutesInterval = %u\n", trigger[i].MinutesInterval);
|
|
TRACE("rgFlags = %u\n", trigger[i].rgFlags);
|
|
TRACE("TriggerType = %u\n", trigger[i].TriggerType);
|
|
TRACE("Reserved2 = %u\n", trigger[i].Reserved2);
|
|
TRACE("wRandomMinutesInterval = %u\n", trigger[i].wRandomMinutesInterval);
|
|
|
|
info->trigger[i] = trigger[i];
|
|
}
|
|
|
|
size -= sizeof(USHORT) + info->trigger_count * sizeof(TASK_TRIGGER);
|
|
data += sizeof(USHORT) + info->trigger_count * sizeof(TASK_TRIGGER);
|
|
|
|
if (size < 2 * sizeof(USHORT) + 64)
|
|
{
|
|
TRACE("no space for signature\n");
|
|
return TRUE; /* signature is optional */
|
|
}
|
|
|
|
signature = (const USHORT *)data;
|
|
TRACE("signature version %04x, client version %04x\n", signature[0], signature[1]);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL load_job(const WCHAR *name, struct job_t *info)
|
|
{
|
|
HANDLE file, mapping;
|
|
DWORD size, try;
|
|
void *data;
|
|
BOOL ret = FALSE;
|
|
|
|
try = 1;
|
|
for (;;)
|
|
{
|
|
file = CreateFileW(name, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, 0);
|
|
if (file == INVALID_HANDLE_VALUE)
|
|
{
|
|
TRACE("Failed to open %s, error %u\n", debugstr_w(name), GetLastError());
|
|
if (GetLastError() != ERROR_SHARING_VIOLATION || try++ >= 3) break;
|
|
Sleep(100);
|
|
continue;
|
|
}
|
|
|
|
size = GetFileSize(file, NULL);
|
|
|
|
mapping = CreateFileMappingW(file, NULL, PAGE_READONLY, 0, 0, 0);
|
|
if (!mapping)
|
|
{
|
|
TRACE("Failed to create file mapping %s, error %u\n", debugstr_w(name), GetLastError());
|
|
CloseHandle(file);
|
|
break;
|
|
}
|
|
|
|
data = MapViewOfFile(mapping, FILE_MAP_READ, 0, 0, 0);
|
|
if (data)
|
|
{
|
|
ret = load_job_data(data, size, info);
|
|
UnmapViewOfFile(data);
|
|
}
|
|
|
|
CloseHandle(mapping);
|
|
CloseHandle(file);
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void free_job_info(AT_ENUM *info)
|
|
{
|
|
heap_free(info->Command);
|
|
}
|
|
|
|
static void free_job(struct job_t *job)
|
|
{
|
|
free_job_info(&job->info);
|
|
heap_free(job->name);
|
|
heap_free(job->params);
|
|
heap_free(job->curdir);
|
|
heap_free(job->trigger);
|
|
heap_free(job);
|
|
}
|
|
|
|
void add_job(const WCHAR *name)
|
|
{
|
|
struct job_t *job;
|
|
|
|
job = heap_alloc_zero(sizeof(*job));
|
|
if (!job) return;
|
|
|
|
if (!load_job(name, job))
|
|
{
|
|
free_job(job);
|
|
return;
|
|
}
|
|
|
|
EnterCriticalSection(&at_job_list_section);
|
|
job->name = heap_strdupW(name);
|
|
job->info.JobId = current_jobid++;
|
|
list_add_tail(&at_job_list, &job->entry);
|
|
LeaveCriticalSection(&at_job_list_section);
|
|
}
|
|
|
|
static inline BOOL is_file(const WIN32_FIND_DATAW *data)
|
|
{
|
|
return !(data->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY);
|
|
}
|
|
|
|
void load_at_tasks(void)
|
|
{
|
|
static const WCHAR tasksW[] = { '\\','T','a','s','k','s','\\',0 };
|
|
static const WCHAR allW[] = { '*',0 };
|
|
WCHAR windir[MAX_PATH], path[MAX_PATH];
|
|
WIN32_FIND_DATAW data;
|
|
HANDLE handle;
|
|
|
|
GetWindowsDirectoryW(windir, MAX_PATH);
|
|
lstrcpyW(path, windir);
|
|
lstrcatW(path, tasksW);
|
|
lstrcatW(path, allW);
|
|
|
|
handle = FindFirstFileW(path, &data);
|
|
if (handle == INVALID_HANDLE_VALUE) return;
|
|
|
|
do
|
|
{
|
|
if (is_file(&data))
|
|
{
|
|
lstrcpyW(path, windir);
|
|
lstrcatW(path, tasksW);
|
|
|
|
if (lstrlenW(path) + lstrlenW(data.cFileName) < MAX_PATH)
|
|
{
|
|
lstrcatW(path, data.cFileName);
|
|
add_job(path);
|
|
}
|
|
else
|
|
FIXME("too long file name %s\n", debugstr_w(data.cFileName));
|
|
}
|
|
} while (FindNextFileW(handle, &data));
|
|
|
|
FindClose(handle);
|
|
}
|
|
|
|
static BOOL write_signature(HANDLE hfile)
|
|
{
|
|
struct
|
|
{
|
|
USHORT SignatureVersion;
|
|
USHORT ClientVersion;
|
|
BYTE md5[64];
|
|
} signature;
|
|
DWORD size;
|
|
|
|
signature.SignatureVersion = 0x0001;
|
|
signature.ClientVersion = 0x0001;
|
|
memset(&signature.md5, 0, sizeof(signature.md5));
|
|
|
|
return WriteFile(hfile, &signature, sizeof(signature), &size, NULL);
|
|
}
|
|
|
|
static BOOL write_reserved_data(HANDLE hfile)
|
|
{
|
|
static const struct
|
|
{
|
|
USHORT size;
|
|
BYTE data[8];
|
|
} user = { 8, { 0xff,0x0f,0x1d,0,0,0,0,0 } };
|
|
DWORD size;
|
|
|
|
return WriteFile(hfile, &user, sizeof(user), &size, NULL);
|
|
}
|
|
|
|
static BOOL write_trigger(HANDLE hfile, const AT_INFO *info)
|
|
{
|
|
USHORT count;
|
|
DWORD size;
|
|
SYSTEMTIME st;
|
|
TASK_TRIGGER trigger;
|
|
|
|
count = 1;
|
|
if (!WriteFile(hfile, &count, sizeof(count), &size, NULL))
|
|
return FALSE;
|
|
|
|
GetSystemTime(&st);
|
|
if (!(info->Flags & JOB_ADD_CURRENT_DATE))
|
|
{
|
|
/* FIXME: parse AT_INFO */
|
|
}
|
|
|
|
trigger.cbTriggerSize = sizeof(trigger);
|
|
trigger.Reserved1 = 0;
|
|
trigger.wBeginYear = st.wYear;
|
|
trigger.wBeginMonth = st.wMonth;
|
|
trigger.wBeginDay = st.wDay;
|
|
trigger.wEndYear = st.wYear;
|
|
trigger.wEndMonth = st.wMonth;
|
|
trigger.wEndDay = st.wDay;
|
|
trigger.wStartHour = st.wHour;
|
|
trigger.wStartMinute = st.wMinute;
|
|
trigger.MinutesDuration = 0;
|
|
trigger.MinutesInterval = 0;
|
|
/* FIXME */
|
|
trigger.rgFlags = TASK_TRIGGER_FLAG_HAS_END_DATE;
|
|
trigger.TriggerType = TASK_TIME_TRIGGER_MONTHLYDATE;
|
|
trigger.Type.MonthlyDate.rgfDays = 0;
|
|
trigger.Type.MonthlyDate.rgfMonths = 0xffff;
|
|
trigger.Reserved2 = 0;
|
|
trigger.wRandomMinutesInterval = 0;
|
|
|
|
return WriteFile(hfile, &trigger, sizeof(trigger), &size, NULL);
|
|
}
|
|
|
|
static BOOL write_unicode_string(HANDLE hfile, const WCHAR *str)
|
|
{
|
|
USHORT count;
|
|
DWORD size;
|
|
|
|
count = str ? (lstrlenW(str) + 1) : 0;
|
|
if (!WriteFile(hfile, &count, sizeof(count), &size, NULL))
|
|
return FALSE;
|
|
|
|
if (!str) return TRUE;
|
|
|
|
count *= sizeof(WCHAR);
|
|
return WriteFile(hfile, str, count, &size, NULL);
|
|
}
|
|
|
|
static BOOL create_job(const WCHAR *job_name, const AT_INFO *info)
|
|
{
|
|
static WCHAR authorW[] = { 'W','i','n','e',0 };
|
|
static WCHAR commentW[] = { 'C','r','e','a','t','e','d',' ','b','y',' ','W','i','n','e',0 };
|
|
FIXDLEN_DATA fixed;
|
|
USHORT word;
|
|
HANDLE hfile;
|
|
DWORD size, ver;
|
|
BOOL ret = FALSE;
|
|
|
|
TRACE("trying to create job %s\n", debugstr_w(job_name));
|
|
hfile = CreateFileW(job_name, GENERIC_WRITE, 0, NULL, CREATE_NEW, 0, 0);
|
|
if (hfile == INVALID_HANDLE_VALUE)
|
|
return FALSE;
|
|
|
|
ver = GetVersion();
|
|
fixed.product_version = MAKEWORD(ver >> 8, ver);
|
|
fixed.file_version = 0x0001;
|
|
UuidCreate(&fixed.uuid);
|
|
fixed.name_size_offset = sizeof(fixed) + sizeof(USHORT); /* FIXDLEN_DATA + Instance Count */
|
|
fixed.trigger_offset = sizeof(fixed) + sizeof(USHORT); /* FIXDLEN_DATA + Instance Count */
|
|
fixed.trigger_offset += sizeof(USHORT) + (lstrlenW(info->Command) + 1) * sizeof(WCHAR); /* Application Name */
|
|
fixed.trigger_offset += sizeof(USHORT); /* Parameters */
|
|
fixed.trigger_offset += sizeof(USHORT); /* Working Directory */
|
|
fixed.trigger_offset += sizeof(USHORT) + (lstrlenW(authorW) + 1) * sizeof(WCHAR); /* Author */
|
|
fixed.trigger_offset += sizeof(USHORT) + (lstrlenW(commentW) + 1) * sizeof(WCHAR); /* Comment */
|
|
fixed.trigger_offset += sizeof(USHORT); /* User Data */
|
|
fixed.trigger_offset += 10; /* Reserved Data */
|
|
fixed.error_retry_count = 0;
|
|
fixed.error_retry_interval = 0;
|
|
fixed.idle_deadline = 60;
|
|
fixed.idle_wait = 10;
|
|
fixed.priority = NORMAL_PRIORITY_CLASS;
|
|
fixed.maximum_runtime = 259200000;
|
|
fixed.exit_code = 0;
|
|
fixed.status = SCHED_S_TASK_HAS_NOT_RUN;
|
|
fixed.flags = TASK_FLAG_DELETE_WHEN_DONE;
|
|
if (!(info->Flags & JOB_NONINTERACTIVE))
|
|
fixed.flags |= TASK_FLAG_INTERACTIVE;
|
|
/* FIXME: add other flags */
|
|
memset(&fixed.last_runtime, 0, sizeof(fixed.last_runtime));
|
|
|
|
if (!WriteFile(hfile, &fixed, sizeof(fixed), &size, NULL))
|
|
goto failed;
|
|
|
|
/* Instance Count */
|
|
word = 0;
|
|
if (!WriteFile(hfile, &word, sizeof(word), &size, NULL))
|
|
goto failed;
|
|
/* Application Name */
|
|
if (!write_unicode_string(hfile, info->Command))
|
|
goto failed;
|
|
/* Parameters */
|
|
if (!write_unicode_string(hfile, NULL))
|
|
goto failed;
|
|
/* Working Directory */
|
|
if (!write_unicode_string(hfile, NULL))
|
|
goto failed;
|
|
/* Author */
|
|
if (!write_unicode_string(hfile, authorW))
|
|
goto failed;
|
|
/* Comment */
|
|
if (!write_unicode_string(hfile, commentW))
|
|
goto failed;
|
|
|
|
/* User Data */
|
|
word = 0;
|
|
if (!WriteFile(hfile, &word, sizeof(word), &size, NULL))
|
|
goto failed;
|
|
|
|
/* Reserved Data */
|
|
if (!write_reserved_data(hfile))
|
|
goto failed;
|
|
|
|
/* Trigegrs */
|
|
if (!write_trigger(hfile, info))
|
|
goto failed;
|
|
|
|
/* Signature */
|
|
if (!write_signature(hfile))
|
|
goto failed;
|
|
|
|
ret = TRUE;
|
|
|
|
failed:
|
|
CloseHandle(hfile);
|
|
if (!ret) DeleteFileW(job_name);
|
|
return ret;
|
|
}
|
|
|
|
static struct job_t *find_job(DWORD jobid, const WCHAR *name, const UUID *id)
|
|
{
|
|
struct job_t *job;
|
|
|
|
LIST_FOR_EACH_ENTRY(job, &at_job_list, struct job_t, entry)
|
|
{
|
|
if (job->info.JobId == jobid || (name && !lstrcmpiW(job->name, name)) || (id && IsEqualGUID(&job->data.uuid, id)))
|
|
return job;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void update_job_status(struct job_t *job)
|
|
{
|
|
HANDLE hfile;
|
|
DWORD try, size;
|
|
#include "pshpack2.h"
|
|
struct
|
|
{
|
|
UINT exit_code;
|
|
UINT status;
|
|
UINT flags;
|
|
SYSTEMTIME last_runtime;
|
|
WORD instance_count;
|
|
} state;
|
|
#include "poppack.h"
|
|
|
|
try = 1;
|
|
for (;;)
|
|
{
|
|
hfile = CreateFileW(job->name, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, 0);
|
|
if (hfile != INVALID_HANDLE_VALUE) break;
|
|
|
|
if (GetLastError() != ERROR_SHARING_VIOLATION || try++ >= 3)
|
|
{
|
|
TRACE("Failed to update %s, error %u\n", debugstr_w(job->name), GetLastError());
|
|
return;
|
|
}
|
|
Sleep(100);
|
|
}
|
|
|
|
if (SetFilePointer(hfile, FIELD_OFFSET(FIXDLEN_DATA, exit_code), NULL, FILE_BEGIN) != INVALID_SET_FILE_POINTER)
|
|
{
|
|
state.exit_code = job->data.exit_code;
|
|
state.status = job->data.status;
|
|
state.flags = job->data.flags;
|
|
state.last_runtime = job->data.last_runtime;
|
|
state.instance_count = job->instance_count;
|
|
WriteFile(hfile, &state, sizeof(state), &size, NULL);
|
|
}
|
|
|
|
CloseHandle(hfile);
|
|
}
|
|
|
|
void update_process_status(DWORD pid)
|
|
{
|
|
struct running_job_t *runjob;
|
|
|
|
EnterCriticalSection(&at_job_list_section);
|
|
|
|
LIST_FOR_EACH_ENTRY(runjob, &running_job_list, struct running_job_t, entry)
|
|
{
|
|
if (runjob->pid == pid)
|
|
{
|
|
struct job_t *job = find_job(0, NULL, &runjob->uuid);
|
|
if (job)
|
|
{
|
|
DWORD exit_code = STILL_ACTIVE;
|
|
|
|
GetExitCodeProcess(runjob->process, &exit_code);
|
|
|
|
if (exit_code != STILL_ACTIVE)
|
|
{
|
|
CloseHandle(runjob->process);
|
|
list_remove(&runjob->entry);
|
|
heap_free(runjob);
|
|
|
|
job->data.exit_code = exit_code;
|
|
job->data.status = SCHED_S_TASK_TERMINATED;
|
|
job->data.flags &= ~0x0c000000;
|
|
job->instance_count = 0;
|
|
update_job_status(job);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
LeaveCriticalSection(&at_job_list_section);
|
|
}
|
|
|
|
void check_task_state(void)
|
|
{
|
|
struct job_t *job;
|
|
struct running_job_t *runjob;
|
|
|
|
EnterCriticalSection(&at_job_list_section);
|
|
|
|
LIST_FOR_EACH_ENTRY(job, &at_job_list, struct job_t, entry)
|
|
{
|
|
if (job->data.flags & 0x08000000)
|
|
{
|
|
TRACE("terminating process %s\n", debugstr_w(job->info.Command));
|
|
|
|
LIST_FOR_EACH_ENTRY(runjob, &running_job_list, struct running_job_t, entry)
|
|
{
|
|
if (IsEqualGUID(&job->data.uuid, &runjob->uuid))
|
|
{
|
|
TerminateProcess(runjob->process, 0);
|
|
update_process_status(runjob->pid);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else if (job->data.flags & 0x04000000)
|
|
{
|
|
STARTUPINFOW si;
|
|
PROCESS_INFORMATION pi;
|
|
|
|
TRACE("running process %s\n", debugstr_w(job->info.Command));
|
|
|
|
if (job->instance_count)
|
|
FIXME("process %s is already running\n", debugstr_w(job->info.Command));
|
|
|
|
runjob = heap_alloc(sizeof(*runjob));
|
|
if (runjob)
|
|
{
|
|
static WCHAR winsta0[] = { 'W','i','n','S','t','a','0',0 };
|
|
|
|
memset(&si, 0, sizeof(si));
|
|
si.cb = sizeof(si);
|
|
/* FIXME: if (job->data.flags & TASK_FLAG_INTERACTIVE) */
|
|
si.lpDesktop = winsta0;
|
|
si.dwFlags = STARTF_USESHOWWINDOW;
|
|
si.wShowWindow = SW_SHOWNORMAL;
|
|
TRACE("executing %s %s at %s\n", debugstr_w(job->info.Command), debugstr_w(job->params), debugstr_w(job->curdir));
|
|
if (CreateProcessW(job->info.Command, job->params, NULL, NULL, FALSE, 0, NULL, job->curdir, &si, &pi))
|
|
{
|
|
CloseHandle(pi.hThread);
|
|
|
|
GetLocalTime(&job->data.last_runtime);
|
|
job->data.exit_code = 0;
|
|
job->data.status = SCHED_S_TASK_RUNNING;
|
|
job->instance_count = 1;
|
|
|
|
runjob->uuid = job->data.uuid;
|
|
runjob->process = pi.hProcess;
|
|
runjob->pid = pi.dwProcessId;
|
|
list_add_tail(&running_job_list, &runjob->entry);
|
|
add_process_to_queue(pi.hProcess);
|
|
}
|
|
else
|
|
{
|
|
WARN("failed to execute %s\n", debugstr_w(job->info.Command));
|
|
job->data.status = SCHED_S_TASK_HAS_NOT_RUN;
|
|
job->instance_count = 0;
|
|
}
|
|
}
|
|
|
|
job->data.flags &= ~0x0c000000;
|
|
update_job_status(job);
|
|
}
|
|
}
|
|
|
|
LeaveCriticalSection(&at_job_list_section);
|
|
}
|
|
|
|
static void run_job(struct job_t *job)
|
|
{
|
|
job->data.flags |= 0x04000000;
|
|
update_job_status(job);
|
|
}
|
|
|
|
void check_task_time(void)
|
|
{
|
|
FILETIME current_ft, begin_ft, end_ft;
|
|
struct job_t *job;
|
|
|
|
GetSystemTimeAsFileTime(¤t_ft);
|
|
FileTimeToLocalFileTime(¤t_ft, ¤t_ft);
|
|
|
|
/* Give -1/+1 minute margin */
|
|
begin_ft = current_ft;
|
|
filetime_add_minutes(&begin_ft, -1);
|
|
end_ft = current_ft;
|
|
filetime_add_minutes(&end_ft, 1);
|
|
|
|
EnterCriticalSection(&at_job_list_section);
|
|
|
|
LIST_FOR_EACH_ENTRY(job, &at_job_list, struct job_t, entry)
|
|
{
|
|
if (job_runs_at(job, &begin_ft, &end_ft))
|
|
{
|
|
run_job(job);
|
|
}
|
|
}
|
|
|
|
LeaveCriticalSection(&at_job_list_section);
|
|
}
|
|
|
|
void check_missed_task_time(void)
|
|
{
|
|
FILETIME current_ft, last_ft;
|
|
struct job_t *job;
|
|
|
|
GetSystemTimeAsFileTime(¤t_ft);
|
|
FileTimeToLocalFileTime(¤t_ft, ¤t_ft);
|
|
|
|
EnterCriticalSection(&at_job_list_section);
|
|
|
|
LIST_FOR_EACH_ENTRY(job, &at_job_list, struct job_t, entry)
|
|
{
|
|
if (SystemTimeToFileTime(&job->data.last_runtime, &last_ft))
|
|
{
|
|
if (job_runs_at(job, &last_ft, ¤t_ft))
|
|
{
|
|
run_job(job);
|
|
}
|
|
}
|
|
}
|
|
|
|
LeaveCriticalSection(&at_job_list_section);
|
|
}
|
|
|
|
void remove_job(const WCHAR *name)
|
|
{
|
|
struct job_t *job;
|
|
|
|
EnterCriticalSection(&at_job_list_section);
|
|
job = find_job(0, name, NULL);
|
|
if (job)
|
|
{
|
|
list_remove(&job->entry);
|
|
free_job(job);
|
|
}
|
|
LeaveCriticalSection(&at_job_list_section);
|
|
}
|
|
|
|
DWORD __cdecl NetrJobAdd(ATSVC_HANDLE server_name, AT_INFO *info, DWORD *jobid)
|
|
{
|
|
WCHAR windir[MAX_PATH];
|
|
|
|
TRACE("%s,%p,%p\n", debugstr_w(server_name), info, jobid);
|
|
|
|
GetWindowsDirectoryW(windir, MAX_PATH);
|
|
|
|
for (;;)
|
|
{
|
|
static const WCHAR fmtW[] = { '\\','T','a','s','k','s','\\','A','t','%','u','.','j','o','b',0 };
|
|
WCHAR task_name[MAX_PATH], name[32];
|
|
|
|
strcpyW(task_name, windir);
|
|
sprintfW(name, fmtW, current_jobid);
|
|
strcatW(task_name, name);
|
|
if (create_job(task_name, info))
|
|
{
|
|
struct job_t *job;
|
|
int i;
|
|
|
|
for (i = 0; i < 5; i++)
|
|
{
|
|
EnterCriticalSection(&at_job_list_section);
|
|
job = find_job(0, task_name, NULL);
|
|
LeaveCriticalSection(&at_job_list_section);
|
|
|
|
if (job)
|
|
{
|
|
*jobid = job->info.JobId;
|
|
break;
|
|
}
|
|
|
|
Sleep(50);
|
|
}
|
|
|
|
if (!job)
|
|
{
|
|
ERR("couldn't find just created job %s\n", debugstr_w(task_name));
|
|
return ERROR_FILE_NOT_FOUND;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
if (GetLastError() != ERROR_FILE_EXISTS)
|
|
{
|
|
|
|
TRACE("create_job error %u\n", GetLastError());
|
|
return GetLastError();
|
|
}
|
|
|
|
InterlockedIncrement(¤t_jobid);
|
|
}
|
|
|
|
return ERROR_SUCCESS;
|
|
}
|
|
|
|
DWORD __cdecl NetrJobDel(ATSVC_HANDLE server_name, DWORD min_jobid, DWORD max_jobid)
|
|
{
|
|
DWORD jobid, ret = APE_AT_ID_NOT_FOUND;
|
|
|
|
TRACE("%s,%u,%u\n", debugstr_w(server_name), min_jobid, max_jobid);
|
|
|
|
EnterCriticalSection(&at_job_list_section);
|
|
|
|
for (jobid = min_jobid; jobid <= max_jobid; jobid++)
|
|
{
|
|
struct job_t *job = find_job(jobid, NULL, NULL);
|
|
|
|
if (!job)
|
|
{
|
|
TRACE("job %u not found\n", jobid);
|
|
ret = APE_AT_ID_NOT_FOUND;
|
|
break;
|
|
}
|
|
|
|
TRACE("deleting job %s\n", debugstr_w(job->name));
|
|
if (!DeleteFileW(job->name))
|
|
{
|
|
ret = GetLastError();
|
|
break;
|
|
}
|
|
|
|
ret = ERROR_SUCCESS;
|
|
}
|
|
|
|
LeaveCriticalSection(&at_job_list_section);
|
|
return ret;
|
|
}
|
|
|
|
static void free_container(AT_ENUM_CONTAINER *container)
|
|
{
|
|
DWORD i;
|
|
|
|
for (i = 0; i < container->EntriesRead; i++)
|
|
heap_free(container->Buffer[i].Command);
|
|
|
|
heap_free(container->Buffer);
|
|
}
|
|
|
|
DWORD __cdecl NetrJobEnum(ATSVC_HANDLE server_name, AT_ENUM_CONTAINER *container,
|
|
DWORD max_length, DWORD *total, DWORD *resume)
|
|
{
|
|
DWORD allocated;
|
|
struct job_t *job;
|
|
|
|
TRACE("%s,%p,%u,%p,%p\n", debugstr_w(server_name), container, max_length, total, resume);
|
|
|
|
*total = 0;
|
|
*resume = 0;
|
|
container->EntriesRead = 0;
|
|
|
|
allocated = 64;
|
|
container->Buffer = heap_alloc(allocated * sizeof(AT_ENUM));
|
|
if (!container->Buffer) return ERROR_NOT_ENOUGH_MEMORY;
|
|
|
|
EnterCriticalSection(&at_job_list_section);
|
|
|
|
LIST_FOR_EACH_ENTRY(job, &at_job_list, struct job_t, entry)
|
|
{
|
|
if (container->EntriesRead >= max_length)
|
|
{
|
|
*resume = container->EntriesRead;
|
|
break;
|
|
}
|
|
|
|
if (allocated <= container->EntriesRead)
|
|
{
|
|
AT_ENUM *new_buffer;
|
|
|
|
allocated *= 2;
|
|
new_buffer = heap_realloc(container->Buffer, allocated * sizeof(AT_ENUM));
|
|
if (!new_buffer)
|
|
{
|
|
free_container(container);
|
|
LeaveCriticalSection(&at_job_list_section);
|
|
return ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
container->Buffer = new_buffer;
|
|
}
|
|
|
|
container->Buffer[container->EntriesRead] = job->info;
|
|
container->Buffer[container->EntriesRead].Command = heap_strdupW(job->info.Command);
|
|
container->EntriesRead++;
|
|
}
|
|
|
|
LeaveCriticalSection(&at_job_list_section);
|
|
|
|
*total = container->EntriesRead;
|
|
|
|
return ERROR_SUCCESS;
|
|
}
|
|
|
|
DWORD __cdecl NetrJobGetInfo(ATSVC_HANDLE server_name, DWORD jobid, AT_INFO **info)
|
|
{
|
|
struct job_t *job;
|
|
DWORD ret = APE_AT_ID_NOT_FOUND;
|
|
|
|
TRACE("%s,%u,%p\n", debugstr_w(server_name), jobid, info);
|
|
|
|
EnterCriticalSection(&at_job_list_section);
|
|
|
|
job = find_job(jobid, NULL, NULL);
|
|
if (job)
|
|
{
|
|
AT_INFO *info_ret = heap_alloc(sizeof(*info_ret));
|
|
if (!info_ret)
|
|
ret = ERROR_NOT_ENOUGH_MEMORY;
|
|
else
|
|
{
|
|
info_ret->JobTime = job->info.JobTime;
|
|
info_ret->DaysOfMonth = job->info.DaysOfMonth;
|
|
info_ret->DaysOfWeek = job->info.DaysOfWeek;
|
|
info_ret->Flags = job->info.Flags;
|
|
info_ret->Command = heap_strdupW(job->info.Command);
|
|
*info = info_ret;
|
|
ret = ERROR_SUCCESS;
|
|
}
|
|
}
|
|
|
|
LeaveCriticalSection(&at_job_list_section);
|
|
return ret;
|
|
}
|