📄 spoolss_win.c
字号:
/* Unix SMB/CIFS implementation. test suite for spoolss rpc operations as performed by various win versions Copyright (C) Kai Blin 2007 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 3 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 "includes.h"#include "torture/torture.h"#include "torture/rpc/rpc.h"#include "librpc/gen_ndr/ndr_spoolss_c.h"#include "rpc_server/dcerpc_server.h"#include "ntvfs/ntvfs.h"#include "param/param.h"struct test_spoolss_win_context { /* EnumPrinters */ uint32_t printer_count; union spoolss_PrinterInfo *printer_info; union spoolss_PrinterInfo *current_info; /* EnumPrinterKeys */ char *printer_keys;};/* This is a convenience function for all OpenPrinterEx calls */static bool test_OpenPrinterEx(struct torture_context *tctx, struct dcerpc_pipe *p, struct policy_handle *handle, const char *printer_name, uint32_t access_mask){ NTSTATUS status; struct spoolss_OpenPrinterEx op; struct spoolss_UserLevel1 ul_1; torture_comment(tctx, "Opening printer '%s'\n", printer_name); op.in.printername = talloc_strdup(tctx, printer_name); op.in.datatype = NULL; op.in.devmode_ctr.devmode = NULL; op.in.access_mask = access_mask; op.in.level = 1; op.in.userlevel.level1 = &ul_1; op.out.handle = handle; ul_1.size = 1234; ul_1.client = "\\clientname"; ul_1.user = "username"; ul_1.build = 1; ul_1.major = 2; ul_1.minor = 3; ul_1.processor = 4567; status = dcerpc_spoolss_OpenPrinterEx(p, tctx, &op); torture_assert_ntstatus_ok(tctx, status, "OpenPrinterEx failed"); torture_assert_werr_ok(tctx, op.out.result, "OpenPrinterEx failed"); return true;}static bool test_OpenPrinterAsAdmin(struct torture_context *tctx, struct dcerpc_pipe *p, const char *printername){ NTSTATUS status; struct spoolss_OpenPrinterEx op; struct spoolss_ClosePrinter cp; struct spoolss_UserLevel1 ul_1; struct policy_handle handle; ul_1.size = 1234; ul_1.client = "\\clientname"; ul_1.user = "username"; ul_1.build = 1; ul_1.major = 2; ul_1.minor = 3; ul_1.processor = 4567; op.in.printername = talloc_strdup(tctx, printername); op.in.datatype = NULL; op.in.devmode_ctr.devmode = NULL; op.in.access_mask = SERVER_ALL_ACCESS; op.in.level = 1; op.in.userlevel.level1 = &ul_1; op.out.handle = &handle; cp.in.handle = &handle; cp.out.handle = &handle; torture_comment(tctx, "Testing OpenPrinterEx(%s) with admin rights\n", op.in.printername); status = dcerpc_spoolss_OpenPrinterEx(p, tctx, &op); if (NT_STATUS_IS_OK(status) && W_ERROR_IS_OK(op.out.result)) { status = dcerpc_spoolss_ClosePrinter(p, tctx, &cp); torture_assert_ntstatus_ok(tctx, status, "ClosePrinter failed"); } return true;}static bool test_ClosePrinter(struct torture_context *tctx, struct dcerpc_pipe *p, struct policy_handle *handle);/* This replicates the opening sequence of OpenPrinterEx calls XP does */static bool test_OpenPrinterSequence(struct torture_context *tctx, struct dcerpc_pipe *p, struct policy_handle *handle){ bool ret; char *printername = talloc_asprintf(tctx, "\\\\%s", dcerpc_server_name(p)); /* First, see if we can open the printer read_only */ ret = test_OpenPrinterEx(tctx, p, handle, printername, 0); torture_assert(tctx, ret == true, "OpenPrinterEx failed."); ret = test_ClosePrinter(tctx, p, handle); torture_assert(tctx, ret, "ClosePrinter failed"); /* Now let's see if we have admin rights to it. */ ret = test_OpenPrinterAsAdmin(tctx, p, printername); torture_assert(tctx, ret == true, "OpenPrinterEx as admin failed unexpectedly."); ret = test_OpenPrinterEx(tctx, p, handle, printername, SERVER_EXECUTE); torture_assert(tctx, ret == true, "OpenPrinterEx failed."); return true;}static bool test_GetPrinterData(struct torture_context *tctx, struct dcerpc_pipe *p, struct policy_handle *handle, const char *value_name, WERROR expected_werr, uint32_t expected_value){ NTSTATUS status; struct spoolss_GetPrinterData gpd; torture_comment(tctx, "Testing GetPrinterData(%s).\n", value_name); gpd.in.handle = handle; gpd.in.value_name = value_name; gpd.in.offered = 4; status = dcerpc_spoolss_GetPrinterData(p, tctx, &gpd); torture_assert_ntstatus_ok(tctx, status, "GetPrinterData failed."); torture_assert_werr_equal(tctx, gpd.out.result, expected_werr, "GetPrinterData did not return expected error value."); if (W_ERROR_IS_OK(expected_werr)) { torture_assert_int_equal(tctx, gpd.out.data.value, expected_value, "GetPrinterData did not return expected value."); } return true;}static bool test_EnumPrinters(struct torture_context *tctx, struct dcerpc_pipe *p, struct test_spoolss_win_context *ctx, uint32_t initial_blob_size){ NTSTATUS status; struct spoolss_EnumPrinters ep; DATA_BLOB blob = data_blob_talloc_zero(ctx, initial_blob_size); ep.in.flags = PRINTER_ENUM_NAME; ep.in.server = talloc_asprintf(tctx, "\\\\%s", dcerpc_server_name(p)); ep.in.level = 2; ep.in.buffer = &blob; ep.in.offered = initial_blob_size; status = dcerpc_spoolss_EnumPrinters(p, ctx, &ep); torture_assert_ntstatus_ok(tctx, status, "EnumPrinters failed."); if (W_ERROR_EQUAL(ep.out.result, WERR_INSUFFICIENT_BUFFER)) { blob = data_blob_talloc_zero(ctx, ep.out.needed); ep.in.buffer = &blob; ep.in.offered = ep.out.needed; status = dcerpc_spoolss_EnumPrinters(p, ctx, &ep); torture_assert_ntstatus_ok(tctx, status,"EnumPrinters failed."); } torture_assert_werr_ok(tctx, ep.out.result, "EnumPrinters failed."); ctx->printer_count = ep.out.count; ctx->printer_info = ep.out.info; torture_comment(tctx, "Found %d printer(s).\n", ctx->printer_count); return true;}static bool test_GetPrinter(struct torture_context *tctx, struct dcerpc_pipe *p, struct policy_handle *handle, struct test_spoolss_win_context *ctx, uint32_t level, uint32_t initial_blob_size){ NTSTATUS status; struct spoolss_GetPrinter gp; DATA_BLOB blob = data_blob_talloc_zero(ctx, initial_blob_size); torture_comment(tctx, "Test GetPrinter level %d\n", level); gp.in.handle = handle; gp.in.level = level; gp.in.buffer = (initial_blob_size == 0)?NULL:&blob; gp.in.offered = initial_blob_size; status = dcerpc_spoolss_GetPrinter(p, tctx, &gp); torture_assert_ntstatus_ok(tctx, status, "GetPrinter failed"); if (W_ERROR_EQUAL(gp.out.result, WERR_INSUFFICIENT_BUFFER)) { blob = data_blob_talloc_zero(ctx, gp.out.needed); gp.in.buffer = &blob; gp.in.offered = gp.out.needed; status = dcerpc_spoolss_GetPrinter(p, tctx, &gp); torture_assert_ntstatus_ok(tctx, status, "GetPrinter failed"); } torture_assert_werr_ok(tctx, gp.out.result, "GetPrinter failed"); ctx->current_info = gp.out.info; return true;}static bool test_EnumJobs(struct torture_context *tctx, struct dcerpc_pipe *p, struct policy_handle *handle){ NTSTATUS status; struct spoolss_EnumJobs ej; DATA_BLOB blob = data_blob_talloc_zero(tctx, 1024); torture_comment(tctx, "Test EnumJobs\n"); ej.in.handle = handle; ej.in.level = 2; ej.in.buffer = &blob; ej.in.offered = 1024; status = dcerpc_spoolss_EnumJobs(p, tctx, &ej); torture_assert_ntstatus_ok(tctx, status, "EnumJobs failed"); torture_assert_werr_ok(tctx, ej.out.result, "EnumJobs failed"); return true;}static bool test_GetPrinterDriver2(struct torture_context *tctx, struct dcerpc_pipe *p, struct policy_handle *handle){ NTSTATUS status; struct spoolss_GetPrinterDriver2 gpd2; DATA_BLOB blob = data_blob_talloc_zero(tctx, 87424); torture_comment(tctx, "Testing GetPrinterDriver2\n"); gpd2.in.handle = handle; gpd2.in.architecture = "Windows NT x86"; gpd2.in.level = 101; gpd2.in.buffer = &blob; gpd2.in.offered = 87424; gpd2.in.client_major_version = 3; gpd2.in.client_minor_version = 0; status = dcerpc_spoolss_GetPrinterDriver2(p, tctx, &gpd2); torture_assert_ntstatus_ok(tctx, status, "GetPrinterDriver2 failed"); torture_assert_werr_ok(tctx, gpd2.out.result,
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -