/* Unix SMB/Netbios implementation. Version 2.2 RPC pipe client Copyright (C) Gerald Carter 2001, Copyright (C) Tim Potter 2000, Copyright (C) Andrew Tridgell 1994-2000 Copyright (C) Luke Kenneth Casson Leighton 1996-2000 Copyright (C) Jean-Francois Micouleau 1999-2000 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, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include "includes.h" extern pstring global_myname; /* Opens a SMB connection to the SPOOLSS pipe */ struct cli_state *cli_spoolss_initialise(struct cli_state *cli, char *system_name, struct ntuser_creds *creds) { return cli_pipe_initialise(cli, system_name, PIPE_SPOOLSS, creds); } /* Open printer ex */ NTSTATUS cli_spoolss_open_printer_ex( struct cli_state *cli, TALLOC_CTX *mem_ctx, char *printername, char *datatype, uint32 access_required, char *station, char *username, POLICY_HND *pol ) { prs_struct qbuf, rbuf; SPOOL_Q_OPEN_PRINTER_EX q; SPOOL_R_OPEN_PRINTER_EX r; NTSTATUS result; ZERO_STRUCT(q); ZERO_STRUCT(r); /* Initialise parse structures */ prs_init(&qbuf, MAX_PDU_FRAG_LEN, mem_ctx, MARSHALL); prs_init(&rbuf, 0, mem_ctx, UNMARSHALL); /* Initialise input parameters */ make_spoolss_q_open_printer_ex(&q, printername, datatype, access_required, station, username); /* Marshall data and send request */ if (!spoolss_io_q_open_printer_ex("", &q, &qbuf, 0) || !rpc_api_pipe_req(cli, SPOOLSS_OPENPRINTEREX, &qbuf, &rbuf)) { result = NT_STATUS_UNSUCCESSFUL; goto done; } /* Unmarshall response */ if (!spoolss_io_r_open_printer_ex("", &r, &rbuf, 0)) { result = NT_STATUS_UNSUCCESSFUL; goto done; } /* Return output parameters */ if (W_ERROR_IS_OK(r.status)) { result = NT_STATUS_OK; *pol = r.handle; } else { result = werror_to_ntstatus(r.status); } done: prs_mem_free(&qbuf); prs_mem_free(&rbuf); return result; } /* Close a printer handle */ NTSTATUS cli_spoolss_close_printer( struct cli_state *cli, TALLOC_CTX *mem_ctx, POLICY_HND *pol ) { prs_struct qbuf, rbuf; SPOOL_Q_CLOSEPRINTER q; SPOOL_R_CLOSEPRINTER r; NTSTATUS result; ZERO_STRUCT(q); ZERO_STRUCT(r); /* Initialise parse structures */ prs_init(&qbuf, MAX_PDU_FRAG_LEN, mem_ctx, MARSHALL); prs_init(&rbuf, 0, mem_ctx, UNMARSHALL); /* Initialise input parameters */ make_spoolss_q_closeprinter(&q, pol); /* Marshall data and send request */ if (!spoolss_io_q_closeprinter("", &q, &qbuf, 0) || !rpc_api_pipe_req(cli, SPOOLSS_CLOSEPRINTER, &qbuf, &rbuf)) { result = NT_STATUS_UNSUCCESSFUL; goto done; } /* Unmarshall response */ if (!spoolss_io_r_closeprinter("", &r, &rbuf, 0)) { result = NT_STATUS_UNSUCCESSFUL; goto done; } /* Return output parameters */ if (W_ERROR_IS_OK(r.status)) { *pol = r.handle; result = NT_STATUS_OK; } else { result = werror_to_ntstatus(r.status); } done: prs_mem_free(&qbuf); prs_mem_free(&rbuf); return result; } /* Initialize a spoolss NEW_BUFFER */ static void init_buffer(NEW_BUFFER *buffer, uint32 size, TALLOC_CTX *ctx) { buffer->ptr = (size != 0); buffer->size = size; buffer->string_at_end = size; prs_init(&buffer->prs, size, ctx, MARSHALL); buffer->struct_start = prs_offset(&buffer->prs); } /* Decode various printer info levels - perhaps this should live in parse_spoolss.c? */ static void decode_printer_info_0( TALLOC_CTX *mem_ctx, NEW_BUFFER *buffer, uint32 returned, PRINTER_INFO_0 **info ) { uint32 i; PRINTER_INFO_0 *inf; inf=(PRINTER_INFO_0 *)talloc(mem_ctx, returned*sizeof(PRINTER_INFO_0)); buffer->prs.data_offset=0; for (i=0; iprs.data_offset=0; for (i=0; iprs.data_offset=0; for (i=0; iprs.data_offset=0; for (i=0; iprs, 0); for (i=0; iprs, 0); for (i=0; iprs.data_offset=0; for (i=0; iprs.data_offset=0; for (i=0; iprs.data_offset=0; for (i=0; iprs, 0); smb_io_driverdir_1("", buffer, inf, 0); *info=inf; } /* Enumerate printers */ NTSTATUS cli_spoolss_enum_printers( struct cli_state *cli, TALLOC_CTX *mem_ctx, uint32 flags, uint32 level, int *returned, PRINTER_INFO_CTR *ctr ) { prs_struct qbuf, rbuf; SPOOL_Q_ENUMPRINTERS q; SPOOL_R_ENUMPRINTERS r; NEW_BUFFER buffer; uint32 needed = 100; NTSTATUS result; fstring server; ZERO_STRUCT(q); ZERO_STRUCT(r); fstrcpy (server, cli->desthost); strupper (server); do { /* Initialise input parameters */ init_buffer(&buffer, needed, mem_ctx); prs_init(&qbuf, MAX_PDU_FRAG_LEN, mem_ctx, MARSHALL); prs_init(&rbuf, 0, mem_ctx, UNMARSHALL); make_spoolss_q_enumprinters(&q, flags, server, level, &buffer, needed); /* Marshall data and send request */ if (!spoolss_io_q_enumprinters("", &q, &qbuf, 0) || !rpc_api_pipe_req(cli, SPOOLSS_ENUMPRINTERS, &qbuf, &rbuf)) { result = NT_STATUS_UNSUCCESSFUL; goto done; } /* Unmarshall response */ if (spoolss_io_r_enumprinters("", &r, &rbuf, 0)) { needed = r.needed; } /* Return output parameters */ if (!W_ERROR_IS_OK(r.status)) { result = werror_to_ntstatus(r.status); goto done; } if ((*returned = r.returned)) { switch (level) { case 1: decode_printer_info_1(mem_ctx, r.buffer, r.returned, &ctr->printers_1); break; case 2: decode_printer_info_2(mem_ctx, r.buffer, r.returned, &ctr->printers_2); break; case 3: decode_printer_info_3(mem_ctx, r.buffer, r.returned, &ctr->printers_3); break; } } done: prs_mem_free(&qbuf); prs_mem_free(&rbuf); } while (NT_STATUS_V(result) == NT_STATUS_V(ERROR_INSUFFICIENT_BUFFER)); return result; } /* Enumerate printer ports */ NTSTATUS cli_spoolss_enum_ports( struct cli_state *cli, TALLOC_CTX *mem_ctx, uint32 level, int *returned, PORT_INFO_CTR *ctr ) { prs_struct qbuf, rbuf; SPOOL_Q_ENUMPORTS q; SPOOL_R_ENUMPORTS r; NEW_BUFFER buffer; uint32 needed = 100; NTSTATUS result; fstring server; ZERO_STRUCT(q); ZERO_STRUCT(r); slprintf (server, sizeof(fstring)-1, "\\\\%s", cli->desthost); strupper (server); do { /* Initialise input parameters */ init_buffer(&buffer, needed, mem_ctx); prs_init(&qbuf, MAX_PDU_FRAG_LEN, mem_ctx, MARSHALL); prs_init(&rbuf, 0, mem_ctx, UNMARSHALL); make_spoolss_q_enumports(&q, server, level, &buffer, needed); /* Marshall data and send request */ if (!spoolss_io_q_enumports("", &q, &qbuf, 0) || !rpc_api_pipe_req(cli, SPOOLSS_ENUMPORTS, &qbuf, &rbuf)) { result = NT_STATUS_UNSUCCESSFUL; goto done; } /* Unmarshall response */ if (spoolss_io_r_enumports("", &r, &rbuf, 0)) { needed = r.needed; } /* Return output parameters */ result = werror_to_ntstatus(r.status); if (NT_STATUS_IS_OK(result) && r.returned > 0) { *returned = r.returned; switch (level) { case 1: decode_port_info_1(mem_ctx, r.buffer, r.returned, &ctr->port.info_1); break; case 2: decode_port_info_2(mem_ctx, r.buffer, r.returned, &ctr->port.info_2); break; } } done: prs_mem_free(&qbuf); prs_mem_free(&rbuf); } while (NT_STATUS_V(result) == NT_STATUS_V(ERROR_INSUFFICIENT_BUFFER)); return result; } /* Get printer info */ NTSTATUS cli_spoolss_getprinter( struct cli_state *cli, TALLOC_CTX *mem_ctx, POLICY_HND *pol, uint32 level, PRINTER_INFO_CTR *ctr ) { prs_struct qbuf, rbuf; SPOOL_Q_GETPRINTER q; SPOOL_R_GETPRINTER r; NEW_BUFFER buffer; uint32 needed = 100; NTSTATUS result; ZERO_STRUCT(q); ZERO_STRUCT(r); do { /* Initialise input parameters */ init_buffer(&buffer, needed, mem_ctx); prs_init(&qbuf, MAX_PDU_FRAG_LEN, mem_ctx, MARSHALL); prs_init(&rbuf, 0, mem_ctx, UNMARSHALL); make_spoolss_q_getprinter(mem_ctx, &q, pol, level, &buffer, needed); /* Marshall data and send request */ if (!spoolss_io_q_getprinter("", &q, &qbuf, 0) || !rpc_api_pipe_req(cli, SPOOLSS_GETPRINTER, &qbuf, &rbuf)) { result = NT_STATUS_UNSUCCESSFUL; goto done; } /* Unmarshall response */ if (spoolss_io_r_getprinter("", &r, &rbuf, 0)) { needed = r.needed; } /* Return output parameters */ result = werror_to_ntstatus(r.status); if (NT_STATUS_IS_OK(result)) { switch (level) { case 0: decode_printer_info_0(mem_ctx, r.buffer, 1, &ctr->printers_0); break; case 1: decode_printer_info_1(mem_ctx, r.buffer, 1, &ctr->printers_1); break; case 2: decode_printer_info_2(mem_ctx, r.buffer, 1, &ctr->printers_2); break; case 3: decode_printer_info_3(mem_ctx, r.buffer, 1, &ctr->printers_3); break; } } done: prs_mem_free(&qbuf); prs_mem_free(&rbuf); } while (NT_STATUS_V(result) == NT_STATUS_V(ERROR_INSUFFICIENT_BUFFER)); return result; } /********************************************************************** * Set printer info */ NTSTATUS cli_spoolss_setprinter( struct cli_state *cli, TALLOC_CTX *mem_ctx, POLICY_HND *pol, uint32 level, PRINTER_INFO_CTR *ctr, uint32 command ) { prs_struct qbuf, rbuf; SPOOL_Q_SETPRINTER q; SPOOL_R_SETPRINTER r; NTSTATUS result = NT_STATUS_ACCESS_DENIED; ZERO_STRUCT(q); ZERO_STRUCT(r); /* Initialise input parameters */ prs_init(&qbuf, MAX_PDU_FRAG_LEN, mem_ctx, MARSHALL); prs_init(&rbuf, 0, mem_ctx, UNMARSHALL); make_spoolss_q_setprinter(mem_ctx, &q, pol, level, ctr, command); /* Marshall data and send request */ if (!spoolss_io_q_setprinter("", &q, &qbuf, 0) || !rpc_api_pipe_req(cli, SPOOLSS_SETPRINTER, &qbuf, &rbuf)) { result = NT_STATUS_ACCESS_DENIED; goto done; } /* Unmarshall response */ if (!spoolss_io_r_setprinter("", &r, &rbuf, 0)) { goto done; } result = werror_to_ntstatus(r.status); done: prs_mem_free(&qbuf); prs_mem_free(&rbuf); return result; } /********************************************************************** * Get installed printer drivers for a given printer */ NTSTATUS cli_spoolss_getprinterdriver ( struct cli_state *cli, TALLOC_CTX *mem_ctx, POLICY_HND *pol, uint32 level, char* env, PRINTER_DRIVER_CTR *ctr ) { prs_struct qbuf, rbuf; SPOOL_Q_GETPRINTERDRIVER2 q; SPOOL_R_GETPRINTERDRIVER2 r; NEW_BUFFER buffer; uint32 needed = 1024; NTSTATUS result; fstring server; ZERO_STRUCT(q); ZERO_STRUCT(r); fstrcpy (server, cli->desthost); strupper (server); do { /* Initialise input parameters */ init_buffer(&buffer, needed, mem_ctx); prs_init(&qbuf, MAX_PDU_FRAG_LEN, mem_ctx, MARSHALL); prs_init(&rbuf, 0, mem_ctx, UNMARSHALL); /* write the request */ make_spoolss_q_getprinterdriver2(&q, pol, env, level, 2, 2, &buffer, needed); /* Marshall data and send request */ if (!spoolss_io_q_getprinterdriver2 ("", &q, &qbuf, 0) || !rpc_api_pipe_req (cli, SPOOLSS_GETPRINTERDRIVER2, &qbuf, &rbuf)) { result = NT_STATUS_UNSUCCESSFUL; goto done; } /* Unmarshall response */ if (spoolss_io_r_getprinterdriver2 ("", &r, &rbuf, 0)) { needed = r.needed; } /* Return output parameters */ result = werror_to_ntstatus(r.status); if (NT_STATUS_IS_OK(result)) { switch (level) { case 1: decode_printer_driver_1(mem_ctx, r.buffer, 1, &ctr->info1); break; case 2: decode_printer_driver_2(mem_ctx, r.buffer, 1, &ctr->info2); break; case 3: decode_printer_driver_3(mem_ctx, r.buffer, 1, &ctr->info3); break; } } done: prs_mem_free(&qbuf); prs_mem_free(&rbuf); } while (NT_STATUS_V(result) == NT_STATUS_V(ERROR_INSUFFICIENT_BUFFER)); return result; } /********************************************************************** * Get installed printer drivers for a given printer */ NTSTATUS cli_spoolss_enumprinterdrivers ( struct cli_state *cli, TALLOC_CTX *mem_ctx, uint32 level, char* env, uint32 *returned, PRINTER_DRIVER_CTR *ctr ) { prs_struct qbuf, rbuf; SPOOL_Q_ENUMPRINTERDRIVERS q; SPOOL_R_ENUMPRINTERDRIVERS r; NEW_BUFFER buffer; uint32 needed = 0; NTSTATUS result; fstring server; ZERO_STRUCT(q); ZERO_STRUCT(r); slprintf (server, sizeof(fstring)-1, "\\\\%s", cli->desthost); strupper (server); do { /* Initialise input parameters */ init_buffer(&buffer, needed, mem_ctx); prs_init(&qbuf, MAX_PDU_FRAG_LEN, mem_ctx, MARSHALL); prs_init(&rbuf, 0, mem_ctx, UNMARSHALL); /* write the request */ make_spoolss_q_enumprinterdrivers(&q, server, env, level, &buffer, needed); /* Marshall data and send request */ if (!spoolss_io_q_enumprinterdrivers ("", &q, &qbuf, 0) || !rpc_api_pipe_req (cli, SPOOLSS_ENUMPRINTERDRIVERS, &qbuf, &rbuf)) { result = NT_STATUS_UNSUCCESSFUL; goto done; } /* Unmarshall response */ if (spoolss_io_r_enumprinterdrivers ("", &r, &rbuf, 0)) { needed = r.needed; } /* Return output parameters */ result = werror_to_ntstatus(r.status); if (NT_STATUS_IS_OK(result) && (r.returned != 0)) { *returned = r.returned; switch (level) { case 1: decode_printer_driver_1(mem_ctx, r.buffer, r.returned, &ctr->info1); break; case 2: decode_printer_driver_2(mem_ctx, r.buffer, r.returned, &ctr->info2); break; case 3: decode_printer_driver_3(mem_ctx, r.buffer, r.returned, &ctr->info3); break; } } done: prs_mem_free(&qbuf); prs_mem_free(&rbuf); } while (NT_STATUS_V(result) == NT_STATUS_V(ERROR_INSUFFICIENT_BUFFER)); return result; } /********************************************************************** * Get installed printer drivers for a given printer */ NTSTATUS cli_spoolss_getprinterdriverdir ( struct cli_state *cli, TALLOC_CTX *mem_ctx, uint32 level, char* env, DRIVER_DIRECTORY_CTR *ctr ) { prs_struct qbuf, rbuf; SPOOL_Q_GETPRINTERDRIVERDIR q; SPOOL_R_GETPRINTERDRIVERDIR r; NEW_BUFFER buffer; uint32 needed = 100; NTSTATUS result; fstring server; ZERO_STRUCT(q); ZERO_STRUCT(r); slprintf (server, sizeof(fstring)-1, "\\\\%s", cli->desthost); strupper (server); do { /* Initialise input parameters */ init_buffer(&buffer, needed, mem_ctx); prs_init(&qbuf, MAX_PDU_FRAG_LEN, mem_ctx, MARSHALL); prs_init(&rbuf, 0, mem_ctx, UNMARSHALL); /* write the request */ make_spoolss_q_getprinterdriverdir(&q, server, env, level, &buffer, needed); /* Marshall data and send request */ if (!spoolss_io_q_getprinterdriverdir ("", &q, &qbuf, 0) || !rpc_api_pipe_req (cli, SPOOLSS_GETPRINTERDRIVERDIRECTORY, &qbuf, &rbuf)) { result = NT_STATUS_UNSUCCESSFUL; goto done; } /* Unmarshall response */ if (spoolss_io_r_getprinterdriverdir ("", &r, &rbuf, 0)) { needed = r.needed; } /* Return output parameters */ result = werror_to_ntstatus(r.status); if (NT_STATUS_IS_OK(result)) { switch (level) { case 1: decode_printerdriverdir_1(mem_ctx, r.buffer, 1, &ctr->info1); break; } } done: prs_mem_free(&qbuf); prs_mem_free(&rbuf); } while (NT_STATUS_V(result) == NT_STATUS_V(ERROR_INSUFFICIENT_BUFFER)); return result; } /********************************************************************** * Install a printer driver */ NTSTATUS cli_spoolss_addprinterdriver ( struct cli_state *cli, TALLOC_CTX *mem_ctx, uint32 level, PRINTER_DRIVER_CTR *ctr ) { prs_struct qbuf, rbuf; SPOOL_Q_ADDPRINTERDRIVER q; SPOOL_R_ADDPRINTERDRIVER r; NTSTATUS result = NT_STATUS_UNSUCCESSFUL; fstring server; ZERO_STRUCT(q); ZERO_STRUCT(r); slprintf (server, sizeof(fstring)-1, "\\\\%s", cli->desthost); strupper (server); /* Initialise input parameters */ prs_init(&qbuf, MAX_PDU_FRAG_LEN, mem_ctx, MARSHALL); prs_init(&rbuf, 0, mem_ctx, UNMARSHALL); /* write the request */ make_spoolss_q_addprinterdriver (mem_ctx, &q, server, level, ctr); /* Marshall data and send request */ result = NT_STATUS_UNSUCCESSFUL; if (!spoolss_io_q_addprinterdriver ("", &q, &qbuf, 0) || !rpc_api_pipe_req (cli, SPOOLSS_ADDPRINTERDRIVER, &qbuf, &rbuf)) { goto done; } /* Unmarshall response */ result = NT_STATUS_UNSUCCESSFUL; if (!spoolss_io_r_addprinterdriver ("", &r, &rbuf, 0)) { goto done; } /* Return output parameters */ result = werror_to_ntstatus(r.status); done: prs_mem_free(&qbuf); prs_mem_free(&rbuf); return result; } /********************************************************************** * Install a printer */ NTSTATUS cli_spoolss_addprinterex ( struct cli_state *cli, TALLOC_CTX *mem_ctx, uint32 level, PRINTER_INFO_CTR *ctr ) { prs_struct qbuf, rbuf; SPOOL_Q_ADDPRINTEREX q; SPOOL_R_ADDPRINTEREX r; NTSTATUS result; fstring server, client, user; ZERO_STRUCT(q); ZERO_STRUCT(r); slprintf (client, sizeof(fstring)-1, "\\\\%s", cli->desthost); strupper (client); slprintf (server, sizeof(fstring)-1, "\\\\%s", cli->desthost); strupper (server); fstrcpy (user, cli->user_name); /* Initialise input parameters */ prs_init(&qbuf, MAX_PDU_FRAG_LEN, mem_ctx, MARSHALL); prs_init(&rbuf, 0, mem_ctx, UNMARSHALL); /* write the request */ make_spoolss_q_addprinterex (mem_ctx, &q, server, client, user, level, ctr); /* Marshall data and send request */ result = NT_STATUS_UNSUCCESSFUL; if (!spoolss_io_q_addprinterex ("", &q, &qbuf, 0) || !rpc_api_pipe_req (cli, SPOOLSS_ADDPRINTEREX, &qbuf, &rbuf)) { goto done; } /* Unmarshall response */ result = NT_STATUS_UNSUCCESSFUL; if (!spoolss_io_r_addprinterex ("", &r, &rbuf, 0)) { goto done; } /* Return output parameters */ result = werror_to_ntstatus(r.status); done: prs_mem_free(&qbuf); prs_mem_free(&rbuf); return result; } /********************************************************************** * Delete a Printer Driver from the server (does not remove * the driver files */ NTSTATUS cli_spoolss_deleteprinterdriver ( struct cli_state *cli, TALLOC_CTX *mem_ctx, char *arch, char *driver ) { prs_struct qbuf, rbuf; SPOOL_Q_DELETEPRINTERDRIVER q; SPOOL_R_DELETEPRINTERDRIVER r; NTSTATUS result; fstring server; ZERO_STRUCT(q); ZERO_STRUCT(r); /* Initialise input parameters */ prs_init(&qbuf, MAX_PDU_FRAG_LEN, mem_ctx, MARSHALL); prs_init(&rbuf, 0, mem_ctx, UNMARSHALL); slprintf (server, sizeof(fstring)-1, "\\\\%s", cli->desthost); strupper (server); /* write the request */ make_spoolss_q_deleteprinterdriver (mem_ctx, &q, server, arch, driver); /* Marshall data and send request */ result = NT_STATUS_UNSUCCESSFUL; if (!spoolss_io_q_deleteprinterdriver ("", &q, &qbuf, 0) || !rpc_api_pipe_req (cli,SPOOLSS_DELETEPRINTERDRIVER , &qbuf, &rbuf)) { goto done; } /* Unmarshall response */ result = NT_STATUS_UNSUCCESSFUL; if (!spoolss_io_r_deleteprinterdriver ("", &r, &rbuf, 0)) { goto done; } /* Return output parameters */ result = werror_to_ntstatus(r.status); done: prs_mem_free(&qbuf); prs_mem_free(&rbuf); return result; } NTSTATUS cli_spoolss_getprintprocessordirectory(struct cli_state *cli, TALLOC_CTX *mem_ctx, char *name, char *environment, fstring procdir) { prs_struct qbuf, rbuf; SPOOL_Q_GETPRINTPROCESSORDIRECTORY q; SPOOL_R_GETPRINTPROCESSORDIRECTORY r; NTSTATUS result; int level = 1; NEW_BUFFER buffer; uint32 needed = 100; ZERO_STRUCT(q); ZERO_STRUCT(r); /* Initialise parse structures */ /* Initialise input parameters */ do { init_buffer(&buffer, needed, mem_ctx); prs_init(&qbuf, MAX_PDU_FRAG_LEN, mem_ctx, MARSHALL); prs_init(&rbuf, 0, mem_ctx, UNMARSHALL); make_spoolss_q_getprintprocessordirectory(&q, name, environment, level, &buffer, needed); /* Marshall data and send request */ if (!spoolss_io_q_getprintprocessordirectory("", &q, &qbuf, 0) || !rpc_api_pipe_req(cli, SPOOLSS_GETPRINTPROCESSORDIRECTORY, &qbuf, &rbuf)) { result = NT_STATUS_UNSUCCESSFUL; goto done; } /* Unmarshall response */ if (!spoolss_io_r_getprintprocessordirectory("", &r, &rbuf, 0)) { result = NT_STATUS_UNSUCCESSFUL; goto done; } /* Return output parameters */ result = werror_to_ntstatus(r.status); } while (NT_STATUS_V(result) == NT_STATUS_V(ERROR_INSUFFICIENT_BUFFER)); done: prs_mem_free(&qbuf); prs_mem_free(&rbuf); return result; }