2 Unix SMB/CIFS implementation.
4 Copyright (C) Andrew Tridgell 1994-1998
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 3 of the License, or
9 (at your option) any later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with this program. If not, see <http://www.gnu.org/licenses/>.
22 /*****************************************************************************
23 Convert a character pointer in a cli_call_api() response to a form we can use.
24 This function contains code to prevent core dumps if the server returns
26 *****************************************************************************/
27 static const char *fix_char_ptr(unsigned int datap, unsigned int converter,
28 char *rdata, int rdrcnt)
30 if (datap == 0) { /* turn NULL pointers into zero length strings */
33 unsigned int offset = datap - converter;
35 if (offset >= rdrcnt) {
36 DEBUG(1,("bad char ptr: datap=%u, converter=%u rdrcnt=%d>",
37 datap, converter, rdrcnt));
40 return &rdata[offset];
46 /****************************************************************************
47 call fn() on each entry in a print queue
48 ****************************************************************************/
49 int cli_print_queue(struct cli_state *cli,
50 void (*fn)(struct print_job_info *))
55 unsigned int rdrcnt, rprcnt;
60 memset(param,'\0',sizeof(param));
63 SSVAL(p,0,76); /* API function number 76 (DosPrintJobEnum) */
65 pstrcpy_base(p,"zWrLeh", param); /* parameter description? */
66 p = skip_string(param,sizeof(param),p);
67 pstrcpy_base(p,"WWzWWDDzz", param); /* returned data format */
68 p = skip_string(param,sizeof(param),p);
69 pstrcpy_base(p,cli->share, param); /* name of queue */
70 p = skip_string(param,sizeof(param),p);
71 SSVAL(p,0,2); /* API function level 2, PRJINFO_2 data structure */
72 SSVAL(p,2,1000); /* size of bytes of returned data buffer */
74 pstrcpy_base(p,"", param); /* subformat */
75 p = skip_string(param,sizeof(param),p);
77 DEBUG(4,("doing cli_print_queue for %s\n", cli->share));
80 param, PTR_DIFF(p,param), 1024, /* Param, length, maxlen */
81 NULL, 0, CLI_BUFFER_SIZE, /* data, length, maxlen */
82 &rparam, &rprcnt, /* return params, length */
83 &rdata, &rdrcnt)) { /* return data, length */
85 result_code = SVAL(rparam,0);
86 converter = SVAL(rparam,2); /* conversion factor */
88 if (result_code == 0) {
89 struct print_job_info job;
93 for (i = 0; i < SVAL(rparam,4); ++i) {
95 job.priority = SVAL(p,2);
97 fix_char_ptr(SVAL(p,4), converter,
99 job.t = cli_make_unix_date3(cli, p + 12);
100 job.size = IVAL(p,16);
101 fstrcpy(job.name,fix_char_ptr(SVAL(p,24),
110 /* If any parameters or data were returned, free the storage. */
117 /****************************************************************************
119 ****************************************************************************/
120 int cli_printjob_del(struct cli_state *cli, int job)
125 unsigned int rdrcnt,rprcnt;
129 memset(param,'\0',sizeof(param));
132 SSVAL(p,0,81); /* DosPrintJobDel() */
134 pstrcpy_base(p,"W", param);
135 p = skip_string(param,sizeof(param),p);
136 pstrcpy_base(p,"", param);
137 p = skip_string(param,sizeof(param),p);
142 param, PTR_DIFF(p,param), 1024, /* Param, length, maxlen */
143 NULL, 0, CLI_BUFFER_SIZE, /* data, length, maxlen */
144 &rparam, &rprcnt, /* return params, length */
145 &rdata, &rdrcnt)) { /* return data, length */
146 ret = SVAL(rparam,0);
156 /****************************************************************************
158 ****************************************************************************/
160 int cli_spl_open(struct cli_state *cli, const char *fname, int flags, int share_mode)
164 unsigned accessmode=0;
168 if (!(flags & O_EXCL)) {
175 accessmode = (share_mode<<4);
177 if ((flags & O_ACCMODE) == O_RDWR) {
179 } else if ((flags & O_ACCMODE) == O_WRONLY) {
184 if ((flags & O_SYNC) == O_SYNC) {
185 accessmode |= (1<<14);
189 if (share_mode == DENY_FCB) {
193 memset(cli->outbuf,'\0',smb_size);
194 memset(cli->inbuf,'\0',smb_size);
196 set_message(NULL,cli->outbuf,15,0,True);
198 SCVAL(cli->outbuf,smb_com,SMBsplopen);
199 SSVAL(cli->outbuf,smb_tid,cli->cnum);
200 cli_setup_packet(cli);
202 SSVAL(cli->outbuf,smb_vwv0,0xFF);
203 SSVAL(cli->outbuf,smb_vwv2,0); /* no additional info */
204 SSVAL(cli->outbuf,smb_vwv3,accessmode);
205 SSVAL(cli->outbuf,smb_vwv4,aSYSTEM | aHIDDEN);
206 SSVAL(cli->outbuf,smb_vwv5,0);
207 SSVAL(cli->outbuf,smb_vwv8,openfn);
209 if (cli->use_oplocks) {
210 /* if using oplocks then ask for a batch oplock via
211 core and extended methods */
212 SCVAL(cli->outbuf,smb_flg, CVAL(cli->outbuf,smb_flg)|
213 FLAG_REQUEST_OPLOCK|FLAG_REQUEST_BATCH_OPLOCK);
214 SSVAL(cli->outbuf,smb_vwv2,SVAL(cli->outbuf,smb_vwv2) | 6);
217 p = smb_buf(cli->outbuf);
218 p += clistr_push(cli, p, fname, -1, STR_TERMINATE);
220 cli_setup_bcc(cli, p);
223 if (!cli_receive_smb(cli)) {
227 if (cli_is_error(cli)) {
231 return SVAL(cli->inbuf,smb_vwv2);
234 /****************************************************************************
236 ****************************************************************************/
238 BOOL cli_spl_close(struct cli_state *cli, int fnum)
240 memset(cli->outbuf,'\0',smb_size);
241 memset(cli->inbuf,'\0',smb_size);
243 set_message(NULL,cli->outbuf,3,0,True);
245 SCVAL(cli->outbuf,smb_com,SMBsplclose);
246 SSVAL(cli->outbuf,smb_tid,cli->cnum);
247 cli_setup_packet(cli);
249 SSVAL(cli->outbuf,smb_vwv0,fnum);
250 SIVALS(cli->outbuf,smb_vwv1,-1);
253 if (!cli_receive_smb(cli)) {
257 return !cli_is_error(cli);