/*
   Unix SMB/CIFS implementation.
   test suite for spoolss rpc operations

   Copyright (C) Gerald (Jerry) Carter 2007
   Copyright (C) Guenther Deschner 2009-2010

   This program is free software; you can redistribute it and/or modify
   it under the terms of the GNU General Public License as published by
   the Free Software Foundation; either version 2 of the License, or
   (at your option) any later version.

   This program 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 General Public License for more details.

   You should have received a copy of the GNU General Public License
   along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/

#include <windows.h>
#include <stdio.h>
#include <assert.h>
#include <sddl.h>

#ifndef MIN
#define MIN(a,b) ((a)<(b)?(a):(b))
#endif

void print_devmode(DEVMODE *pDevModeIn)
{
	if (pDevModeIn == NULL) {
		printf("\tDevice Mode\t= (null)\n");
		return;
	}

	printf("\tDEVMODE:----------\n");
	printf("\tDeviceName:      [%s]\n", pDevModeIn->dmDeviceName);
	printf("\tSpecVersion:     %d\n",   pDevModeIn->dmSpecVersion);
	printf("\tDriverVersion:   %d\n",   pDevModeIn->dmDriverVersion);
	printf("\tSize:            %d\n",   pDevModeIn->dmSize);
	printf("\tDriverExtra:     %d\n",   pDevModeIn->dmDriverExtra);
	printf("\tFields:          0x%x\n", pDevModeIn->dmFields);

	if (pDevModeIn->dmFields & DM_ORIENTATION)
		printf("\tOrientation:     %d\n", pDevModeIn->dmOrientation);
	if (pDevModeIn->dmFields & DM_PAPERSIZE)
		printf("\tPaperSize:       %d\n", pDevModeIn->dmPaperSize);
	if (pDevModeIn->dmFields & DM_PAPERLENGTH)
		printf("\tPaperLength:     %d\n", pDevModeIn->dmPaperLength);
	if (pDevModeIn->dmFields & DM_PAPERWIDTH)
		printf("\tPaperWidth:      %d\n", pDevModeIn->dmPaperWidth);
//	if (pDevModeIn->dmFields & DM_POSITION)
//		printf("\tPosition:        %d\n", pDevModeIn->dmPosition);
	if (pDevModeIn->dmFields & DM_SCALE)
		printf("\tScale:           %d\n", pDevModeIn->dmScale);
	if (pDevModeIn->dmFields & DM_COPIES)
		printf("\tCopies:          %d\n", pDevModeIn->dmCopies );
	if (pDevModeIn->dmFields & DM_DEFAULTSOURCE)
		printf("\tDefaultSource:   %d\n", pDevModeIn->dmDefaultSource);
	if (pDevModeIn->dmFields & DM_PRINTQUALITY)
		printf("\tPrintQuality:    %d\n", pDevModeIn->dmPrintQuality);
	if (pDevModeIn->dmFields & DM_COLOR)
		printf("\tColor:           %d\n", pDevModeIn->dmColor);
	if (pDevModeIn->dmFields & DM_DUPLEX)
		printf("\tDuplex:          %d\n", pDevModeIn->dmDuplex);
	if (pDevModeIn->dmFields & DM_YRESOLUTION)
		printf("\tYResolution:     %d\n", pDevModeIn->dmYResolution);
	if (pDevModeIn->dmFields & DM_TTOPTION)
		printf("\tTTOption:        %d\n", pDevModeIn->dmTTOption);
	if (pDevModeIn->dmFields & DM_COLLATE)
		printf("\tCollate:         %d\n", pDevModeIn->dmCollate);
	if (pDevModeIn->dmFields & DM_FORMNAME)
		printf("\tForm:            [%s]\n", pDevModeIn->dmFormName);
	if (pDevModeIn->dmFields & DM_LOGPIXELS)
		printf("\tLogPixels:       %d\n", pDevModeIn->dmLogPixels);
	if (pDevModeIn->dmFields & DM_BITSPERPEL)
		printf("\tBitsPerPel:      %d\n", pDevModeIn->dmBitsPerPel);
	if (pDevModeIn->dmFields & DM_PELSWIDTH)
		printf("\tPelsWidth:       %d\n", pDevModeIn->dmPelsWidth);
	if (pDevModeIn->dmFields & DM_PELSHEIGHT)
		printf("\tPelsHeight:      %d\n", pDevModeIn->dmPelsHeight);
	if (pDevModeIn->dmFields & DM_DISPLAYFLAGS)
		printf("\tDisplayFlags:    %d\n", pDevModeIn->dmDisplayFlags);
//	if (pDevModeIn->dmFields & DM_NUP)
//		printf("\tNup:             %d\n", pDevModeIn->dmNup);
	if (pDevModeIn->dmFields & DM_DISPLAYFREQUENCY)
		printf("\tDisplayFrequency:%d\n", pDevModeIn->dmDisplayFrequency);
	if (pDevModeIn->dmFields & DM_ICMMETHOD)
		printf("\tICMMethod:       %d\n", pDevModeIn->dmICMMethod);
	if (pDevModeIn->dmFields & DM_ICMINTENT)
		printf("\tICMIntent:       %d\n", pDevModeIn->dmICMIntent);
	if (pDevModeIn->dmFields & DM_MEDIATYPE)
		printf("\tMediaType:       %d\n", pDevModeIn->dmMediaType);
	if (pDevModeIn->dmFields & DM_DITHERTYPE)
		printf("\tDitherType:      %d\n", pDevModeIn->dmDitherType);
//	if (pDevModeIn->dmFields & DM_PANNINGWIDTH)
//		printf("\tPanningWidth:    %d\n", pDevModeIn->dmPanningWidth);
//	if (pDevModeIn->dmFields & DM_PANNINGHEIGHT)
//		printf("\tPanningHeight:   %d\n", pDevModeIn->dmPanningHeight);

#if 0
	if (bForceIn) {
		printf("DEVMODE\n");
		Dump((BYTE*)pDevModeIn, sizeof(DEVMODE), LEADER);

		if (pDevModeIn->dmDriverExtra) {
			printf("DriverExtra\n");
			Dump((BYTE*)pDevModeIn + sizeof(DEVMODE), pDevModeIn->dmDriverExtra, LEADER);
		}
	}
#endif

	return;
}

void print_acl(const char* str, ACL *acl)
{
	printf("%s\n", str);
	if (acl == NULL)
		return;

	printf("\t\tACL Revision \t\t 0x%x\n", acl->AclRevision);
	printf("\t\tSbz1\t\t 0x%x\n", acl->Sbz1);
	printf("\t\tSbz2\t\t 0x%x\n", acl->Sbz2);
	printf("\t\tACL Size\t\t 0x%x\n", acl->AclSize);
	printf("\t\tACL Count\t\t 0x%x\n", acl->AceCount);

	return;
}

void PrintLastError();

void print_sid(LPSTR str, PSID sid)
{
	LPSTR sid_string;

	printf("%s\n", str);

	if (sid == NULL) {
		printf("(null sid)\n");
		return;
	}

	if (!ConvertSidToStringSid(sid, &sid_string)) {
		PrintLastError();
		return;
	}

	printf("%s\n", sid_string);
	LocalFree(sid_string);

	return;
}

void print_secdesc(SECURITY_DESCRIPTOR *secdesc)
{
	LPSTR sd_string;

	if (secdesc == NULL) {
		printf("\tSecurity Descriptor\t= (null)\n");
		return;
	}

	if (!ConvertSecurityDescriptorToStringSecurityDescriptor(secdesc, 1, 7, &sd_string, NULL)) {
		PrintLastError();
		return;
	}

	printf("%s\n", sd_string);
	LocalFree(sd_string);

#if 0
	printf("\tRevision\t= 0x%x\n", secdesc->Revision);
	printf("\tSbz1\t\t= 0x%x\n", secdesc->Sbz1);
	printf("\tControl\t\t= 0x%x\n", secdesc->Control);
	print_sid("\tOwner\t\t= ", secdesc->Owner);
	print_sid("\tGroup\t\t= ",secdesc->Group);
	print_acl("\tSacl\t\t= ", secdesc->Sacl);
	print_acl("\tDacl\t\t= ", secdesc->Dacl);
#endif
	return;
}

void PrintLastError()
{
	LPVOID lpMsgBuf;
	DWORD status;

	status = GetLastError();

	FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
		      NULL, GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
		      (LPTSTR)&lpMsgBuf, 0, NULL);
	printf("ERROR  [0x%x] : %s\n", status, (char*)lpMsgBuf);
	LocalFree(lpMsgBuf);

	return;
}


void print_job_info_1(PJOB_INFO_1 info)
{
	printf("\tJob ID\t\t= %d\n",		info->JobId);
	printf("\tPrinter Name\t= %s\n",	info->pPrinterName);
	printf("\tMachine Name\t= %s\n",	info->pMachineName);
	printf("\tUser Name\t= %s\n",		info->pUserName);
	printf("\tDocument\t= %s\n",		info->pDocument);
	printf("\tDatatype\t= %s\n",		info->pDatatype);
	printf("\tStatus\t\t= %s\n",		info->pStatus);
	printf("\tStatus\t= %d\n",		info->Status);
	printf("\tPriority\t= %d\n",		info->Priority);
	printf("\tPosition\t= %d\n",		info->Position);
	printf("\tTotal Pages\t= %d\n",	info->TotalPages);
	printf("\tPages Printed\t= %d\n",	info->PagesPrinted);
	printf("\tSubmitted (DD:MM:YY HH:MM:SS)\t= %d:%d:%d %d:%d:%d UTC\n",
		info->Submitted.wDay, info->Submitted.wMonth,
		info->Submitted.wYear, info->Submitted.wHour,
		info->Submitted.wMinute, info->Submitted.wSecond);

	return;
}

void print_job_info_2(PJOB_INFO_2 info)
{
	printf("\tJob ID\t\t= %d\n",		info->JobId);
	printf("\tPrinter Name\t= %s\n",	info->pPrinterName);
	printf("\tMachine Name\t= %s\n",	info->pMachineName);
	printf("\tUser Name\t= %s\n",		info->pUserName);
	printf("\tDocument\t= %s\n",		info->pDocument);
	printf("\tDatatype\t= %s\n",		info->pDatatype);
	printf("\tNotify Name\t= %s\n",		info->pNotifyName);
	printf("\tPrint Processor\t= %s\n",	info->pPrintProcessor);
	printf("\tParameters\t= %s\n",		info->pParameters);
	printf("\tDriver Name\t= %s\n",		info->pDriverName);
	printf("\tStatus\t\t= %s\n",		info->pStatus);
	printf("\tStatus\t\t= %d\n",		info->Status);
	printf("\tPriority\t= %d\n",		info->Priority);
	printf("\tPosition\t= %d\n",		info->Position);
	printf("\tTotal Pages\t= %d\n",		info->TotalPages);
	printf("\tPages Printed\t= %d\n",	info->PagesPrinted);
	printf("\tStart Time\t= %d\n",		info->StartTime);
	printf("\tUntil Time\t= %d\n",		info->UntilTime);
	printf("\tTime\t\t= %d\n",		info->Time);
	printf("\tSize\t\t= %d\n",		info->Size);
	printf("\tSubmitted (DD:MM:YY HH:MM:SS)\t= %d:%d:%d %d:%d:%d UTC\n",
		info->Submitted.wDay, info->Submitted.wMonth,
		info->Submitted.wYear, info->Submitted.wHour,
		info->Submitted.wMinute, info->Submitted.wSecond);
	printf("\tDevice Mode Information\n");
	printf("\t-----------------------\n");
	print_devmode(info->pDevMode);
	printf("\tSecurity Descriptor Information\n");
	printf("\t-------------------------------\n");
	print_secdesc(info->pSecurityDescriptor);

	return;
}

void print_job_info_3(PJOB_INFO_3 info)
{
	printf("\tJob ID\t\t= %d\n",		info->JobId);
	printf("\tJob ID Next Job\t= %d\n",	info->NextJobId);
	printf("\tReserved (must be 0)\t= %d\n",info->Reserved);

	return;
}

void print_job_info_4(PJOB_INFO_4 info)
{
	printf("\tJob ID\t\t= %d\n",		info->JobId);
	printf("\tPrinter Name\t= %s\n",	info->pPrinterName);
	printf("\tMachine Name\t= %s\n",	info->pMachineName);
	printf("\tUser Name\t= %s\n",		info->pUserName);
	printf("\tDocument\t= %s\n",		info->pDocument);
	printf("\tDatatype\t= %s\n",		info->pDatatype);
	printf("\tNotify Name\t= %s\n",		info->pNotifyName);
	printf("\tPrint Processor\t= %s\n",	info->pPrintProcessor);
	printf("\tDriver Name\t= %s\n",		info->pDriverName);
	printf("\tStatus\t\t= %s\n",		info->pStatus);
	printf("\tStatus\t\t= %d\n",		info->Status);
	printf("\tPriority\t= %d\n",		info->Priority);
	printf("\tPosition\t= %d\n",		info->Position);
	printf("\tTotal Pages\t= %d\n",		info->TotalPages);
	printf("\tPages Printed\t= %d\n",	info->PagesPrinted);
	printf("\tStart Time\t= %d\n",		info->StartTime);
	printf("\tUntil Time\t= %d\n",		info->UntilTime);
	printf("\tTime\t\t= %d\n",		info->Time);
	printf("\tSize\t\t= %d\n",		info->Size);
	printf("\tSize High\t\t= 0x%016x\n",	info->SizeHigh);
	printf("\tSubmitted (DD:MM:YY HH:MM:SS)\t= %d:%d:%d %d:%d:%d UTC\n",
		info->Submitted.wDay, info->Submitted.wMonth,
		info->Submitted.wYear, info->Submitted.wHour,
		info->Submitted.wMinute, info->Submitted.wSecond);
	printf("\tDevice Mode Information\n");
	printf("\t-----------------------\n");
	print_devmode(info->pDevMode);
	printf("\tSecurity Descriptor Information\n");
	printf("\t-------------------------------\n");
	print_secdesc(info->pSecurityDescriptor);

	return;
}

void print_job_info_bylevel(DWORD level, LPBYTE buffer, DWORD count)
{
	DWORD i;
	PJOB_INFO_1	buffer1 = NULL;
	PJOB_INFO_2	buffer2 = NULL;
	PJOB_INFO_3	buffer3 = NULL;
	PJOB_INFO_4	buffer4 = NULL;

	if (!buffer) {
		return;
	}

	switch (level) {
	case 1:
		buffer1 = (PJOB_INFO_1)buffer;
		break;
	case 2:
		buffer2 = (PJOB_INFO_2)buffer;
		break;
	case 3:
		buffer3 = (PJOB_INFO_3)buffer;
		break;
	case 4:
		buffer4 = (PJOB_INFO_4)buffer;
		break;
	default:
		break;
	}

	printf("Job Info Level %d:\n", level);

	switch (level) {
	case 1:
		for (i=0; i<count; i++) {
			print_job_info_1(&buffer1[i]);
			printf("\n");
		}
		break;
	case 2:
		for (i=0; i<count; i++) {
			print_job_info_2(&buffer2[i]);
			printf("\n");
		}
		break;
	case 3:
		for (i=0; i<count; i++) {
			print_job_info_3(&buffer3[i]);
			printf("\n");
		}
		break;
	case 4:
		for (i=0; i<count; i++) {
			print_job_info_4(&buffer4[i]);
			printf("\n");
		}
		break;
	default:
		break;
	}
}

void print_monitor_info_1(PMONITOR_INFO_1 info)
{
	printf("\tMonitor Name\t= %s\n",	info->pName);

	return;
}

void print_monitor_info_2(PMONITOR_INFO_2 info)
{
	printf("\tMonitor Name\t= %s\n",	info->pName);
	printf("\tEnvironment\t= %s\n",		info->pEnvironment);
	printf("\tDLL Name\t= %s\n",		info->pDLLName);

	return;
}

void print_monitor_info_bylevel(DWORD level, LPBYTE buffer, DWORD count)
{
	DWORD i;
	PMONITOR_INFO_1 buffer1 = NULL;
	PMONITOR_INFO_2 buffer2 = NULL;

	if (!buffer) {
		return;
	}

	switch (level) {
	case 1:
		buffer1 = (PMONITOR_INFO_1)buffer;
		break;
	case 2:
		buffer2 = (PMONITOR_INFO_2)buffer;
		break;
	default:
		break;
	}

	printf("Monitor Info Level %d:\n", level);

	switch (level) {
	case 1:
		for (i=0; i<count; i++) {
			print_monitor_info_1(&buffer1[i]);
			printf("\n");
		}
		break;
	case 2:
		for (i=0; i<count; i++) {
			print_monitor_info_2(&buffer2[i]);
			printf("\n");
		}
		break;
	default:
		break;
	}
}

void print_port_info_1(PPORT_INFO_1 info)
{
	printf("\tPort Name\t= %s\n",	info->pName);
	return;
}

void print_port_info_2(PPORT_INFO_2 info)
{
	printf("\tPort Name\t= %s\n",	info->pPortName);
	printf("\tMonitor Name\t= %s\n",info->pMonitorName);
	printf("\tDescription\t= %s\n",	info->pDescription);
	printf("\tPort Type\t= 0x%08x\n", info->fPortType);
	printf("\tReserved\t= 0x%08x\n", info->Reserved);
	return;
}

void print_port_info_3(PPORT_INFO_3 info)
{
	printf("\tStatus\t= 0x%08x\n", info->dwStatus);
	printf("\tStatus String\t= %s\n", info->pszStatus);
	printf("\tSeverity\t= 0x%08x\n", info->dwSeverity);
	return;
}

void print_port_info_bylevel(DWORD level, LPBYTE buffer, DWORD count)
{
	DWORD i;
	PPORT_INFO_1 buffer1 = NULL;
	PPORT_INFO_2 buffer2 = NULL;
	PPORT_INFO_3 buffer3 = NULL;

	if (!buffer) {
		return;
	}

	switch (level) {
	case 1:
		buffer1 = (PPORT_INFO_1)buffer;
		break;
	case 2:
		buffer2 = (PPORT_INFO_2)buffer;
		break;
	case 3:
		buffer3 = (PPORT_INFO_3)buffer;
		break;
	default:
		break;
	}

	printf("Port Info Level %d:\n", level);

	switch (level) {
	case 1:
		for (i=0; i<count; i++) {
			print_port_info_1(&buffer1[i]);
			printf("\n");
		}
		break;
	case 2:
		for (i=0; i<count; i++) {
			print_port_info_2(&buffer2[i]);
			printf("\n");
		}
		break;
	case 3:
		for (i=0; i<count; i++) {
			print_port_info_3(&buffer3[i]);
			printf("\n");
		}
		break;
	default:
		break;
	}
}

void print_form_info_1(PFORM_INFO_1 info)
{
	printf("\tForm Name\t= %s\n",	info->pName);
	printf("\tFlags\t\t= 0x%x\n",	info->Flags);
	printf("\tSize\t\t= %d x %d\n", info->Size.cx, info->Size.cy);
	printf("\tRectangle\t= [left]%d [right]%d [top]%d [bottom]%d\n",
			info->ImageableArea.left, info->ImageableArea.right,
			info->ImageableArea.top, info->ImageableArea.bottom);

	return;
}

void print_form_info_2(PFORM_INFO_2 info)
{
	printf("\tForm Name\t= %s\n",	info->pName);
	printf("\tFlags\t\t= 0x%x\n",	info->Flags);
	printf("\tSize\t\t= %d x %d\n", info->Size.cx, info->Size.cy);
	printf("\tRectangle\t= [left]%d [right]%d [top]%d [bottom]%d\n",
			info->ImageableArea.left, info->ImageableArea.right,
			info->ImageableArea.top, info->ImageableArea.bottom);
	printf("\tKeyword\t= %s\n",	info->pKeyword);
	printf("\tString Type\t= 0x%08x\n", info->StringType);
	printf("\tMui DLL\t= %s\n",	info->pMuiDll);
	printf("\tResource Id\t= 0x%08x\n", info->dwResourceId);
	printf("\tDisplay Name\t= %s\n",info->pDisplayName);
	printf("\tLang Id\t= 0x%04x\n", info->wLangId);

	return;
}

void print_form_info_bylevel(DWORD level, LPBYTE buffer, DWORD count)
{
	DWORD i;
	PFORM_INFO_1 buffer1 = NULL;
	PFORM_INFO_2 buffer2 = NULL;

	if (!buffer) {
		return;
	}

	switch (level) {
	case 1:
		buffer1 = (PFORM_INFO_1)buffer;
		break;
	case 2:
		buffer2 = (PFORM_INFO_2)buffer;
		break;
	default:
		break;
	}

	printf("Form Info Level %d:\n", level);

	switch (level) {
	case 1:
		for (i=0; i<count; i++) {
			print_form_info_1(&buffer1[i]);
			printf("\n");
		}
		break;
	case 2:
		for (i=0; i<count; i++) {
			print_form_info_2(&buffer2[i]);
			printf("\n");
		}
		break;
	default:
		break;
	}
}

void print_printer_info_1(PPRINTER_INFO_1 info)
{
	printf("\tPrinter Name\t= %s\n",	info->pName);
	printf("\tDescription\t= %s\n",		info->pDescription);
	printf("\tComment\t\t= %s\n",		info->pComment);
	printf("\tFlags\t\t= 0x%x\n",		info->Flags);

	return;
}

void print_printer_info_2(PPRINTER_INFO_2 info)
{
	printf("\tServer Name\t\t= %s\n",	info->pServerName);
	printf("\tPrinter Name\t\t= %s\n",	info->pPrinterName);
	printf("\tPort Name\t\t= %s\n",		info->pPortName);
	printf("\tShare Name\t\t= %s\n",	info->pShareName);
	printf("\tDriver Name\t\t= %s\n",	info->pDriverName);
	printf("\tComment\t\t\t= %s\n",		info->pComment);
	printf("\tLocation\t\t= %s\n",		info->pLocation);
	printf("\tSeparator File\t\t= %s\n",	info->pSepFile);
	printf("\tDefault Data Type\t= %s\n",	info->pDatatype);
	printf("\tPrint Processor\t\t= %s\n",	info->pPrintProcessor);
	printf("\tParameters\t\t= %s\n",	info->pParameters);
	printf("\tAttributes\t\t= 0x%x\n",	info->Attributes);
	printf("\tPriority\t\t= 0x%x\n",	info->Priority);
	printf("\tDefault Priority\t= 0x%x\n",	info->DefaultPriority);
	printf("\tStart Time\t\t= 0x%x\n",	info->StartTime);
	printf("\tUntil Time\t\t= 0x%x\n",	info->UntilTime);
	printf("\tStatus\t\t\t= 0x%x\n",	info->Status);
	printf("\tcJobs\t\t\t= 0x%x\n",		info->cJobs);
	printf("\tAverage PPM\t\t= 0x%x\n",	info->AveragePPM);

	printf("\tDevice Mode Information\n");
	printf("\t-----------------------\n");
	print_devmode(info->pDevMode);
	printf("\tSecurity Descriptor Information\n");
	printf("\t-------------------------------\n");
	print_secdesc(info->pSecurityDescriptor);
	return;
}

void print_printer_info_3(PPRINTER_INFO_3 info)
{
	printf("\tSecurity Descriptor Information\n");
	printf("\t-------------------------------\n");
	print_secdesc(info->pSecurityDescriptor);
	return;
}

void print_printer_info_4(PPRINTER_INFO_4 info)
{
	printf("\tServer Name\t\t= %s\n",	info->pServerName);
	printf("\tPrinter Name\t\t= %s\n",	info->pPrinterName);
	printf("\tAttributes\t\t= 0x%x\n",	info->Attributes);
	return;
}

void print_printer_info_5(PPRINTER_INFO_5 info)
{
	printf("\tPrinter Name\t\t\t= %s\n",		info->pPrinterName);
	printf("\tPort Name\t\t\t= %s\n",		info->pPortName);
	printf("\tAttributes\t\t\t= 0x%x\n",		info->Attributes);
	printf("\tDev NotSelect Timeout\t= 0x%x\n",	info->DeviceNotSelectedTimeout);
	printf("\tTX RetryTimeout\t\t= 0x%x\n",		info->TransmissionRetryTimeout);
	return;
}

void print_printer_info_6(PPRINTER_INFO_6 info)
{
	printf("\tStatus\t\t\t= 0x%x\n",		info->dwStatus);
	return;
}

void print_printer_info_7(PPRINTER_INFO_7 info)
{
	printf("\tObject GUID\t\t\t= %s\n",		info->pszObjectGUID);
	printf("\tAction\t\t\t= 0x%x\n",		info->dwAction);
	return;
}

void print_printer_info_8(PPRINTER_INFO_8 info)
{
	print_devmode(info->pDevMode);
}

void print_printer_info_9(PPRINTER_INFO_9 info)
{
	print_devmode(info->pDevMode);
}

void print_printer_info_bylevel(DWORD level, LPBYTE buffer, DWORD count)
{
	DWORD i;
	PPRINTER_INFO_1	buffer1 = NULL;
	PPRINTER_INFO_2	buffer2 = NULL;
	PPRINTER_INFO_3	buffer3 = NULL;
	PPRINTER_INFO_4	buffer4 = NULL;
	PPRINTER_INFO_5	buffer5 = NULL;
	PPRINTER_INFO_6	buffer6 = NULL;
	PPRINTER_INFO_7	buffer7 = NULL;
	PPRINTER_INFO_8	buffer8 = NULL;
	PPRINTER_INFO_9	buffer9 = NULL;

	if (!buffer) {
		return;
	}

	switch (level) {
	case 1:
		buffer1 = (PPRINTER_INFO_1)buffer;
		break;
	case 2:
		buffer2 = (PPRINTER_INFO_2)buffer;
		break;
	case 3:
		buffer3 = (PPRINTER_INFO_3)buffer;
		break;
	case 4:
		buffer4 = (PPRINTER_INFO_4)buffer;
		break;
	case 5:
		buffer5 = (PPRINTER_INFO_5)buffer;
		break;
	case 6:
		buffer6 = (PPRINTER_INFO_6)buffer;
		break;
	case 7:
		buffer7 = (PPRINTER_INFO_7)buffer;
		break;
	case 8:
		buffer8 = (PPRINTER_INFO_8)buffer;
		break;
	case 9:
		buffer9 = (PPRINTER_INFO_9)buffer;
		break;
	default:
		break;
	}

	printf("Printer Info Level %d:\n", level);

	switch (level) {
	case 1:
		for (i=0; i<count; i++) {
			print_printer_info_1(&buffer1[i]);
			printf("\n");
		}
		break;
	case 2:
		for (i=0; i<count; i++) {
			print_printer_info_2(&buffer2[i]);
			printf("\n");
		}
		break;
	case 3:
		for (i=0; i<count; i++) {
			print_printer_info_3(&buffer3[i]);
			printf("\n");
		}
		break;
	case 4:
		for (i=0; i<count; i++) {
			print_printer_info_4(&buffer4[i]);
			printf("\n");
		}
		break;
	case 5:
		for (i=0; i<count; i++) {
			print_printer_info_5(&buffer5[i]);
			printf("\n");
		}
		break;
	case 6:
		for (i=0; i<count; i++) {
			print_printer_info_6(&buffer6[i]);
			printf("\n");
		}
		break;
	case 7:
		for (i=0; i<count; i++) {
			print_printer_info_7(&buffer7[i]);
			printf("\n");
		}
		break;
	case 8:
		for (i=0; i<count; i++) {
			print_printer_info_8(&buffer8[i]);
			printf("\n");
		}
		break;
	case 9:
		for (i=0; i<count; i++) {
			print_printer_info_9(&buffer9[i]);
			printf("\n");
		}
		break;
	default:
		break;
	}
}

void print_printprocessor_info_1(PPRINTPROCESSOR_INFO_1 info)
{
	printf("\tPrint Processor Name\t= %s\n", info->pName);

	return;
}

void print_printprocessor_info_bylevel(DWORD level, LPBYTE buffer, DWORD count)
{
	DWORD i;
	PPRINTPROCESSOR_INFO_1 buffer1 = NULL;

	if (!buffer) {
		return;
	}

	switch (level) {
	case 1:
		buffer1 = (PPRINTPROCESSOR_INFO_1)buffer;
		break;
	default:
		break;
	}

	printf("Print Processor Info Level %d:\n", level);

	switch (level) {
	case 1:
		for (i=0; i<count; i++) {
			print_printprocessor_info_1(&buffer1[i]);
			printf("\n");
		}
		break;
	default:
		break;
	}
}

void print_datatypes_info_1(PDATATYPES_INFO_1 info)
{
	printf("\tDataTypes Name\t= %s\n", info->pName);

	return;
}

void print_datatypes_info_bylevel(DWORD level, LPBYTE buffer, DWORD count)
{
	DWORD i;
	PDATATYPES_INFO_1 buffer1 = NULL;

	if (!buffer) {
		return;
	}

	switch (level) {
	case 1:
		buffer1 = (PDATATYPES_INFO_1)buffer;
		break;
	default:
		break;
	}

	printf("DataTypes Info Level %d:\n", level);

	switch (level) {
	case 1:
		for (i=0; i<count; i++) {
			print_datatypes_info_1(&buffer1[i]);
			printf("\n");
		}
		break;
	default:
		break;
	}
}

void print_driver_info_1(PDRIVER_INFO_1 info)
{
	printf("\tDriver Name\t= %s\n\n",	info->pName);

	return;
}

void print_driver_info_2(PDRIVER_INFO_2 info)
{
	printf("\tDriver Name\t= %s\n",		info->pName);
	printf("\tEnvironment\t= %s\n",		info->pEnvironment);
	printf("\tVersion\t\t= %d\n",		info->cVersion);
	printf("\tDriver Path\t= %s\n",		info->pDriverPath);
	printf("\tData File\t= %s\n",		info->pDataFile);
	printf("\tConfig File\t= %s\n\n",	info->pConfigFile);

	return;
}

void print_driver_info_3(PDRIVER_INFO_3 info)
{
	char *ptr = NULL;

	printf("\tDriver Name\t= %s\n",		info->pName);
	printf("\tEnvironment\t= %s\n",		info->pEnvironment);
	printf("\tVersion\t\t= %d\n",		info->cVersion);
	printf("\tDriver Path\t= %s\n",		info->pDriverPath);
	printf("\tData File\t= %s\n",		info->pDataFile);
	printf("\tConfig File\t= %s\n",		info->pConfigFile);
	printf("\tHelp Path\t= %s\n",		info->pHelpFile);
	printf("\tMonitor Name\t= %s\n",	info->pMonitorName);
	printf("\tData Type\t= %s\n",		info->pDefaultDataType);
	ptr = (char*)info->pDependentFiles;
	while ((ptr != NULL) && (*ptr != '\0')) {
		printf("\tDependent Files\t= %s\n", ptr);
		for (;*ptr != '\0'; ptr++)
			/* printf("%s\n", ptr); */
			;
		ptr++;
	}

	return;
}

void print_driver_info_4(PDRIVER_INFO_4 info)
{
	char *ptr = NULL;

	printf("\tDriver Name\t= %s\n",		info->pName);
	printf("\tEnvironment\t= %s\n",		info->pEnvironment);
	printf("\tVersion\t\t= %d\n",		info->cVersion);
	printf("\tDriver Path\t= %s\n",		info->pDriverPath);
	printf("\tData File\t= %s\n",		info->pDataFile);
	printf("\tConfig File\t= %s\n",		info->pConfigFile);
	printf("\tHelp Path\t= %s\n",		info->pHelpFile);
	printf("\tMonitor Name\t= %s\n",	info->pMonitorName);
	printf("\tData Type\t= %s\n",		info->pDefaultDataType);
	printf("\tPrevious Names\t= %s\n",	info->pszzPreviousNames);
	ptr = (char*)info->pDependentFiles;
	while ((ptr != NULL) && (*ptr != '\0')) {
		printf("\tDependent Files\t= %s\n", ptr);
		for (;*ptr != '\0'; ptr++)
			/* printf("%s\n", ptr); */
			;
		ptr++;
	}

	return;
}

void print_driver_info_5(PDRIVER_INFO_5 info)
{
	printf("\tDriver Name\t= %s\n",		info->pName);
	printf("\tEnvironment\t= %s\n",		info->pEnvironment);
	printf("\tVersion\t\t= %d\n",		info->cVersion);
	printf("\tDriver Path\t= %s\n",		info->pDriverPath);
	printf("\tData File\t= %s\n",		info->pDataFile);
	printf("\tConfig File\t= %s\n",		info->pConfigFile);
	printf("\tDriver Attributes\t= %d\n",	info->dwDriverAttributes);
	printf("\tConfig Version\t= %d\n",	info->dwConfigVersion);
	printf("\tDriver Version\t= %d\n",	info->dwDriverVersion);

	return;
}

void print_driver_info_6(PDRIVER_INFO_6 info)
{
	char *ptr = NULL;

	printf("\tDriver Name\t= %s\n",		info->pName);
	printf("\tEnvironment\t= %s\n",		info->pEnvironment);
	printf("\tVersion\t\t= %d\n",		info->cVersion);
	printf("\tDriver Path\t= %s\n",		info->pDriverPath);
	printf("\tData File\t= %s\n",		info->pDataFile);
	printf("\tConfig File\t= %s\n",		info->pConfigFile);
	printf("\tHelp Path\t= %s\n",		info->pHelpFile);
	printf("\tMonitor Name\t= %s\n",	info->pMonitorName);
	printf("\tData Type\t= %s\n",		info->pDefaultDataType);
	printf("\tPrevious Names\t= %s\n",	info->pszzPreviousNames);
	ptr = (char*)info->pDependentFiles;
	if (ptr != NULL) {
		while (*ptr != '\0') {
			printf("\tDependent Files\t= %s\n", ptr);
			for (;*ptr != '\0'; ptr++)
				/* printf("%s\n", ptr); */
				;
			ptr++;
		}
	} else {
		printf("\tPrevious Names\t= (null)\n");
	}

	ptr = (char*)info->pszzPreviousNames;
	if (ptr != NULL) {
		while (*ptr != '\0') {
			printf("\tPrevious Names\t= %s\n", ptr);
			for (;*ptr != '\0'; ptr++)
				/* printf("%s\n", ptr); */
				;
			ptr++;
		}
	} else {
		printf("\tPrevious Names\t= (null)\n");
	}

	printf("\tDriver Date\t= %d (%08x:%08x)\n",
		info->ftDriverDate,
		info->ftDriverDate.dwLowDateTime,
		info->ftDriverDate.dwHighDateTime);
	printf("\t\t");
	print_file_time(&info->ftDriverDate);
	printf("\tDriver Version\t= %d\n",	info->dwlDriverVersion);
	printf("\tManufacture Name = %s\n",	info->pszMfgName);
	printf("\tOEM URL\t\t= %s\n",		info->pszOEMUrl);
	printf("\tHardware ID\t= %s\n",		info->pszHardwareID);
	printf("\tProvider\t= %s\n",		info->pszProvider);
	return;
}

static void print_multi_sz(LPSTR multisz)
{
	char *ptr = NULL;

	ptr = (char *)multisz;

	if (!ptr) {
		printf("(null)\n");
		return;
	}

	while (*ptr != '\0') {
		printf("%s\n", ptr);
		for (; *ptr != '\0'; ptr++) {
			/* printf("%s\n", ptr); */
			;
		}
		ptr++;
	}
}

void print_driver_info_8(PDRIVER_INFO_8 info)
{
	printf("\tDriver Name\t= %s\n",		info->pName);
	printf("\tEnvironment\t= %s\n",		info->pEnvironment);
	printf("\tVersion\t\t= %d\n",		info->cVersion);
	printf("\tDriver Path\t= %s\n",		info->pDriverPath);
	printf("\tData File\t= %s\n",		info->pDataFile);
	printf("\tConfig File\t= %s\n",		info->pConfigFile);
	printf("\tHelp Path\t= %s\n",		info->pHelpFile);
	printf("\tMonitor Name\t= %s\n",	info->pMonitorName);
	printf("\tData Type\t= %s\n",		info->pDefaultDataType);
	printf("\tPrevious Names\t=\n");
	print_multi_sz(info->pszzPreviousNames);
	printf("\tDependent Files\t=\n");
	print_multi_sz(info->pDependentFiles);
	printf("\tDriver Date\t= %d (%08x:%08x)\n",
		info->ftDriverDate,
		info->ftDriverDate.dwLowDateTime,
		info->ftDriverDate.dwHighDateTime);
	printf("\t\t");
	print_file_time(&info->ftDriverDate);
	printf("\tDriver Version\t= %d\n",	info->dwlDriverVersion);
	printf("\tManufacture Name = %s\n",	info->pszMfgName);
	printf("\tOEM URL\t\t= %s\n",		info->pszOEMUrl);
	printf("\tHardware ID\t= %s\n",		info->pszHardwareID);
	printf("\tProvider\t= %s\n",		info->pszProvider);
	printf("\tPrint Processor\t= %s\n",	info->pszPrintProcessor);
	printf("\tVendor Setup\t= %s\n",	info->pszVendorSetup);
	printf("\tColor Profiles\t=\n");
	print_multi_sz(info->pszzColorProfiles);
	printf("\tInf Path\t= %s\n",		info->pszInfPath);
	printf("\tPrinter Driver Attributes = %d\n", info->dwPrinterDriverAttributes);
	printf("\tCore Driver Dependencies\t=\n");
	print_multi_sz(info->pszzCoreDriverDependencies);
	printf("\tMin Inbox Driver VerDate\t= %d (%08x:%08x)\n",
		info->ftMinInboxDriverVerDate,
		info->ftMinInboxDriverVerDate.dwLowDateTime,
		info->ftMinInboxDriverVerDate.dwHighDateTime);
	printf("\t\t");
	print_file_time(&info->ftMinInboxDriverVerDate);
	printf("\tMin Inbox Driver VerVersion\t= %d\n", info->dwlMinInboxDriverVerVersion);
	return;
}

void print_driver_info_bylevel(DWORD level, LPBYTE buffer, DWORD count)
{
	DWORD i;
	PDRIVER_INFO_1	buffer1 = NULL;
	PDRIVER_INFO_2	buffer2 = NULL;
	PDRIVER_INFO_3	buffer3 = NULL;
	PDRIVER_INFO_4	buffer4 = NULL;
	PDRIVER_INFO_5	buffer5 = NULL;
	PDRIVER_INFO_6	buffer6 = NULL;
	PDRIVER_INFO_8	buffer8 = NULL;

	if (!buffer) {
		return;
	}

	switch (level) {
	case 1:
		buffer1 = (PDRIVER_INFO_1)buffer;
		break;
	case 2:
		buffer2 = (PDRIVER_INFO_2)buffer;
		break;
	case 3:
		buffer3 = (PDRIVER_INFO_3)buffer;
		break;
	case 4:
		buffer4 = (PDRIVER_INFO_4)buffer;
		break;
	case 5:
		buffer5 = (PDRIVER_INFO_5)buffer;
		break;
	case 6:
		buffer6 = (PDRIVER_INFO_6)buffer;
		break;
	case 8:
		buffer8 = (PDRIVER_INFO_8)buffer;
		break;
	default:
		break;
	}

	printf("Driver Info Level %d:\n", level);

	switch (level) {
	case 1:
		for (i=0; i<count; i++) {
			print_driver_info_1(&buffer1[i]);
			printf("\n");
		}
		break;
	case 2:
		for (i=0; i<count; i++) {
			print_driver_info_2(&buffer2[i]);
			printf("\n");
		}
		break;
	case 3:
		for (i=0; i<count; i++) {
			print_driver_info_3(&buffer3[i]);
			printf("\n");
		}
		break;
	case 4:
		for (i=0; i<count; i++) {
			print_driver_info_4(&buffer4[i]);
			printf("\n");
		}
		break;
	case 5:
		for (i=0; i<count; i++) {
			print_driver_info_5(&buffer5[i]);
			printf("\n");
		}
		break;
	case 6:
		for (i=0; i<count; i++) {
			print_driver_info_6(&buffer6[i]);
			printf("\n");
		}
		break;
	case 8:
		for (i=0; i<count; i++) {
			print_driver_info_8(&buffer8[i]);
			printf("\n");
		}
		break;
	default:
		break;
	}
}

void print_doc_info_1(PDOC_INFO_1 info)
{
	printf("\tDocument Name\t= %s\n",	info->pDocName);
	printf("\tOutput Filename\t= %s\n",	info->pOutputFile);
	printf("\tDatatype\t= %s\n",		info->pDatatype);
	return;
}

void print_printer_keys(LPSTR buffer)
{
	LPSTR p = NULL;

	p = buffer;

	while (p && *p) {
		printf("%s\n", p);
		for (; *p; p = CharNext(p)) {
			p = CharNext(p);
		}
	}
}

LPSTR reg_type_str(DWORD type)
{
	switch (type) {
	case REG_DWORD:
		return "REG_DWORD";
	case REG_SZ:
		return "REG_SZ";
	case REG_MULTI_SZ:
		return "REG_MULTI_SZ";
	case REG_BINARY:
		return "REG_BINARY";
	default:
		return NULL;
	}
}

void print_asc(const BYTE *buf, DWORD len)
{
	int i;
	for (i=0; i<len; i++) {
		printf("%c", isprint(buf[i])?buf[i]:'.');
	}
}

static void dump_data(const BYTE *buf, int len)
{
	int i=0;
	static const BYTE empty[16] = { 0, };

	if (len<=0) return;

	for (i=0; i<len;) {

		if (i%16 == 0) {
			if ((i > 0) &&
			    (len > i+16) &&
			    (memcmp(&buf[i], &empty, 16) == 0))
			{
				i +=16;
				continue;
			}

			if (i<len)  {
				printf("[%04X] ",i);
			}
		}

		printf("%02x ", buf[i]);
		i++;

		if (i%8 == 0) printf("  ");
		if (i%16 == 0) {
			print_asc(&buf[i-16],8); printf(" ");
			print_asc(&buf[i-8],8); printf("\n");
		}
	}

	if (i%16) {
		int n;
		n = 16 - (i%16);
		printf(" ");
		if (n>8) printf(" ");
		while (n--) printf("   ");
		n = MIN(8,i%16);
		print_asc(&buf[i-(i%16)],n); printf( " " );
		n = (i%16) - n;
		if (n>0) print_asc(&buf[i-n],n);
		printf("\n");
	}
}

static void dump_printer_data(DWORD size, LPBYTE buffer, DWORD type)
{
	DWORD i = 0;
	LPSTR p = NULL;

	switch (type) {
	case REG_SZ:
		dump_data(buffer, size);
		break;
	case REG_MULTI_SZ:
		dump_data(buffer, size);
		p = (LPSTR)buffer;
		while (p && *p) {
			printf("\t\t%s\n", p);
			for (; *p; p = CharNext(p)) {
				p = CharNext(p);
			}
		}
		break;
	case REG_DWORD:
		assert(size == 4);
		printf("\t\t0x%08x\n", (DWORD)*buffer);
		break;
	case REG_BINARY:
		dump_data(buffer, size);
		break;
	default:
		break;
	}
}

void print_printer_data(LPSTR keyname, LPSTR valuename, DWORD size, LPBYTE buffer, DWORD type)
{
	if (keyname) {
		printf("\tKey Name:\t%s\n", keyname);
	}

	printf("\tValue Name:\t%s\n", valuename);
	printf("\tSize:\t\t0x%x (%d)\n", size, size);
	printf("\tType:\t\t%s\n", reg_type_str(type));

	if (buffer == NULL || size == 0) {
		return;
	}

	dump_printer_data(size, buffer, type);
}

void print_printer_dataw(LPCWSTR keyname, LPCWSTR valuename, DWORD size, LPBYTE buffer, DWORD type)
{
	if (keyname) {
		printf("\tKey Name:\t%ls\n", keyname);
	}

	printf("\tValue Name:\t%ls\n", valuename);
	printf("\tSize:\t\t0x%x (%d)\n", size, size);
	printf("\tType:\t\t%s\n", reg_type_str(type));

	if (buffer == NULL || size == 0) {
		return;
	}

	dump_printer_data(size, buffer, type);
}


void print_printer_enum_values(PRINTER_ENUM_VALUES *info)
{
	print_printer_data(NULL, info->pValueName, info->cbData, info->pData, info->dwType);
}

void print_file_time(const FILETIME *t)
{
	SYSTEMTIME s;
	LPSTR dayofweek = NULL;
	LPSTR month = NULL;

	if (!FileTimeToSystemTime(t, &s)) {
		printf("Failed to convert FILETIME to SYSTEMTIME\n");
		return;
	}
	switch (s.wDayOfWeek) {
	case 0:
		dayofweek = "Sun";
		break;
	case 1:
		dayofweek = "Mon";
		break;
	case 2:
		dayofweek = "Tue";
		break;
	case 3:
		dayofweek = "Wed";
		break;
	case 4:
		dayofweek = "Thu";
		break;
	case 5:
		dayofweek = "Fri";
		break;
	case 6:
		dayofweek = "Sat";
		break;
	default:
		break;
	}

	switch (s.wMonth) {
	case 1:
		month = "Jan";
		break;
	case 2:
		month = "Feb";
		break;
	case 3:
		month = "Mar";
		break;
	case 4:
		month = "Apr";
		break;
	case 5:
		month = "May";
		break;
	case 6:
		month = "Jun";
		break;
	case 7:
		month = "Jul";
		break;
	case 8:
		month = "Aug";
		break;
	case 9:
		month = "Sep";
		break;
	case 10:
		month = "Oct";
		break;
	case 11:
		month = "Nov";
		break;
	case 12:
		month = "Dec";
		break;
	default:
		break;
	}

	printf("%s %s %02d %02d:%02d:%02d %04d\n",
		dayofweek, month, s.wDay,
		s.wHour, s.wMinute, s.wSecond, s.wYear);
}