2013-12-29 04:53:12 +00:00
|
|
|
/*
|
2016-03-25 19:59:40 +00:00
|
|
|
* Licensed to the Apache Software Foundation (ASF) under one
|
|
|
|
* or more contributor license agreements. See the NOTICE file
|
|
|
|
* distributed with this work for additional information
|
|
|
|
* regarding copyright ownership. The ASF licenses this file
|
|
|
|
* to you under the Apache License, Version 2.0 (the
|
|
|
|
* "License"); you may not use this file except in compliance
|
|
|
|
* with the License. You may obtain a copy of the License at
|
2013-12-29 04:53:12 +00:00
|
|
|
*
|
2016-03-25 19:59:40 +00:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2013-12-29 04:53:12 +00:00
|
|
|
*
|
2016-03-25 19:59:40 +00:00
|
|
|
* Unless required by applicable law or agreed to in writing,
|
|
|
|
* software distributed under the License is distributed on an
|
|
|
|
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
|
|
* KIND, either express or implied. See the License for the
|
|
|
|
* specific language governing permissions and limitations
|
|
|
|
* under the License.
|
2013-12-29 04:53:12 +00:00
|
|
|
*/
|
2013-06-23 00:43:05 +00:00
|
|
|
|
2014-01-01 22:44:28 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "rdpdr_messages.h"
|
|
|
|
#include "rdpdr_printer.h"
|
2017-02-13 22:44:06 +00:00
|
|
|
#include "rdpdr_print_job.h"
|
2014-01-01 22:44:28 +00:00
|
|
|
#include "rdpdr_service.h"
|
|
|
|
#include "rdp_status.h"
|
|
|
|
|
|
|
|
#include <freerdp/utils/svc_plugin.h>
|
2014-06-11 01:45:14 +00:00
|
|
|
#include <guacamole/client.h>
|
2014-01-01 22:44:28 +00:00
|
|
|
#include <guacamole/protocol.h>
|
2016-06-09 20:57:22 +00:00
|
|
|
#include <guacamole/socket.h>
|
|
|
|
#include <guacamole/stream.h>
|
|
|
|
#include <guacamole/user.h>
|
2014-01-01 22:44:28 +00:00
|
|
|
|
2013-07-17 18:45:53 +00:00
|
|
|
#ifdef ENABLE_WINPR
|
|
|
|
#include <winpr/stream.h>
|
|
|
|
#else
|
|
|
|
#include "compat/winpr-stream.h"
|
|
|
|
#endif
|
|
|
|
|
2014-06-11 01:45:14 +00:00
|
|
|
#include <errno.h>
|
2017-02-13 22:44:06 +00:00
|
|
|
#include <pthread.h>
|
2016-06-09 20:57:22 +00:00
|
|
|
#include <stdint.h>
|
2014-06-11 01:45:14 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2013-07-24 22:52:46 +00:00
|
|
|
void guac_rdpdr_process_print_job_create(guac_rdpdr_device* device,
|
|
|
|
wStream* input_stream, int completion_id) {
|
2013-06-23 00:43:05 +00:00
|
|
|
|
2017-02-13 22:44:06 +00:00
|
|
|
/* Log creation of print job */
|
|
|
|
guac_client_log(device->rdpdr->client, GUAC_LOG_INFO, "Print job created");
|
2013-10-22 21:08:03 +00:00
|
|
|
|
2017-02-13 22:44:06 +00:00
|
|
|
/* Create print job */
|
|
|
|
device->data = guac_client_for_owner(device->rdpdr->client,
|
|
|
|
guac_rdpdr_print_job_alloc, NULL);
|
|
|
|
|
|
|
|
/* Respond with success */
|
2013-10-22 21:08:03 +00:00
|
|
|
wStream* output_stream = guac_rdpdr_new_io_completion(device,
|
|
|
|
completion_id, STATUS_SUCCESS, 4);
|
2013-06-23 00:43:05 +00:00
|
|
|
|
2013-07-17 18:45:53 +00:00
|
|
|
Stream_Write_UINT32(output_stream, 0); /* fileId */
|
2013-07-24 22:52:46 +00:00
|
|
|
svc_plugin_send((rdpSvcPlugin*) device->rdpdr, output_stream);
|
2013-06-23 00:43:05 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-07-24 22:52:46 +00:00
|
|
|
void guac_rdpdr_process_print_job_write(guac_rdpdr_device* device,
|
|
|
|
wStream* input_stream, int completion_id) {
|
2013-06-23 00:43:05 +00:00
|
|
|
|
2017-02-13 22:44:06 +00:00
|
|
|
guac_rdpdr_print_job* job = (guac_rdpdr_print_job*) device->data;
|
2013-06-23 00:43:05 +00:00
|
|
|
|
2017-02-13 22:44:06 +00:00
|
|
|
unsigned char* buffer;
|
|
|
|
int length;
|
|
|
|
int status;
|
2013-06-23 00:43:05 +00:00
|
|
|
|
2017-02-13 22:44:06 +00:00
|
|
|
/* Read buffer of print data */
|
2013-07-17 18:45:53 +00:00
|
|
|
Stream_Read_UINT32(input_stream, length);
|
|
|
|
Stream_Seek(input_stream, 8); /* Offset */
|
|
|
|
Stream_Seek(input_stream, 20); /* Padding */
|
|
|
|
buffer = Stream_Pointer(input_stream);
|
2013-06-23 00:43:05 +00:00
|
|
|
|
2017-02-13 22:44:06 +00:00
|
|
|
/* Write data only if job exists, translating status for RDP */
|
|
|
|
if (job != NULL && (length = guac_rdpdr_print_job_write(job,
|
|
|
|
buffer, length)) >= 0) {
|
|
|
|
status = STATUS_SUCCESS;
|
|
|
|
}
|
2013-06-28 18:26:28 +00:00
|
|
|
|
2017-02-13 22:44:06 +00:00
|
|
|
/* Report device offline if write fails */
|
|
|
|
else {
|
|
|
|
status = STATUS_DEVICE_OFF_LINE;
|
|
|
|
length = 0;
|
2013-06-28 18:26:28 +00:00
|
|
|
}
|
|
|
|
|
2017-02-13 22:44:06 +00:00
|
|
|
wStream* output_stream = guac_rdpdr_new_io_completion(device,
|
|
|
|
completion_id, status, 5);
|
2013-06-23 00:43:05 +00:00
|
|
|
|
2013-07-17 18:45:53 +00:00
|
|
|
Stream_Write_UINT32(output_stream, length);
|
2013-10-22 21:08:03 +00:00
|
|
|
Stream_Write_UINT8(output_stream, 0); /* Padding */
|
2013-06-23 00:43:05 +00:00
|
|
|
|
2013-07-24 22:52:46 +00:00
|
|
|
svc_plugin_send((rdpSvcPlugin*) device->rdpdr, output_stream);
|
2013-06-23 00:43:05 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-07-24 22:52:46 +00:00
|
|
|
void guac_rdpdr_process_print_job_close(guac_rdpdr_device* device,
|
|
|
|
wStream* input_stream, int completion_id) {
|
2013-06-23 00:43:05 +00:00
|
|
|
|
2017-02-13 22:44:06 +00:00
|
|
|
/* End print job */
|
|
|
|
guac_rdpdr_print_job* job = (guac_rdpdr_print_job*) device->data;
|
|
|
|
if (job != NULL) {
|
|
|
|
guac_rdpdr_print_job_free(job);
|
|
|
|
device->data = NULL;
|
|
|
|
}
|
2013-10-22 21:08:03 +00:00
|
|
|
|
|
|
|
wStream* output_stream = guac_rdpdr_new_io_completion(device,
|
2016-03-04 23:56:30 +00:00
|
|
|
completion_id, STATUS_SUCCESS, 4);
|
2013-10-22 21:08:03 +00:00
|
|
|
|
2016-03-04 23:56:30 +00:00
|
|
|
Stream_Write_UINT32(output_stream, 0); /* Padding */
|
2017-02-13 22:44:06 +00:00
|
|
|
svc_plugin_send((rdpSvcPlugin*) device->rdpdr, output_stream);
|
2013-06-23 00:43:05 +00:00
|
|
|
|
2017-02-13 22:44:06 +00:00
|
|
|
/* Log end of print job */
|
2014-11-08 00:32:19 +00:00
|
|
|
guac_client_log(device->rdpdr->client, GUAC_LOG_INFO, "Print job closed");
|
2013-07-24 22:52:46 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-07-25 01:28:20 +00:00
|
|
|
static void guac_rdpdr_device_printer_announce_handler(guac_rdpdr_device* device,
|
|
|
|
wStream* output_stream, int device_id) {
|
|
|
|
|
|
|
|
/* Printer header */
|
2014-11-08 00:32:19 +00:00
|
|
|
guac_client_log(device->rdpdr->client, GUAC_LOG_INFO, "Sending printer");
|
2013-07-25 01:28:20 +00:00
|
|
|
Stream_Write_UINT32(output_stream, RDPDR_DTYP_PRINT);
|
|
|
|
Stream_Write_UINT32(output_stream, device_id);
|
|
|
|
Stream_Write(output_stream, "PRN1\0\0\0\0", 8); /* DOS name */
|
|
|
|
|
|
|
|
/* Printer data */
|
|
|
|
Stream_Write_UINT32(output_stream, 24 + GUAC_PRINTER_DRIVER_LENGTH + GUAC_PRINTER_NAME_LENGTH);
|
|
|
|
Stream_Write_UINT32(output_stream,
|
|
|
|
RDPDR_PRINTER_ANNOUNCE_FLAG_DEFAULTPRINTER
|
|
|
|
| RDPDR_PRINTER_ANNOUNCE_FLAG_NETWORKPRINTER);
|
|
|
|
Stream_Write_UINT32(output_stream, 0); /* reserved - must be 0 */
|
|
|
|
Stream_Write_UINT32(output_stream, 0); /* PnPName length (PnPName is ultimately ignored) */
|
|
|
|
Stream_Write_UINT32(output_stream, GUAC_PRINTER_DRIVER_LENGTH); /* DriverName length */
|
|
|
|
Stream_Write_UINT32(output_stream, GUAC_PRINTER_NAME_LENGTH); /* PrinterName length */
|
|
|
|
Stream_Write_UINT32(output_stream, 0); /* CachedFields length */
|
|
|
|
|
|
|
|
Stream_Write(output_stream, GUAC_PRINTER_DRIVER, GUAC_PRINTER_DRIVER_LENGTH);
|
|
|
|
Stream_Write(output_stream, GUAC_PRINTER_NAME, GUAC_PRINTER_NAME_LENGTH);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-07-24 22:52:46 +00:00
|
|
|
static void guac_rdpdr_device_printer_iorequest_handler(guac_rdpdr_device* device,
|
|
|
|
wStream* input_stream, int file_id, int completion_id, int major_func, int minor_func) {
|
|
|
|
|
|
|
|
switch (major_func) {
|
|
|
|
|
|
|
|
/* Print job create */
|
|
|
|
case IRP_MJ_CREATE:
|
|
|
|
guac_rdpdr_process_print_job_create(device, input_stream, completion_id);
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Printer job write */
|
|
|
|
case IRP_MJ_WRITE:
|
|
|
|
guac_rdpdr_process_print_job_write(device, input_stream, completion_id);
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Printer job close */
|
|
|
|
case IRP_MJ_CLOSE:
|
|
|
|
guac_rdpdr_process_print_job_close(device, input_stream, completion_id);
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Log unknown */
|
|
|
|
default:
|
2014-11-08 00:32:19 +00:00
|
|
|
guac_client_log(device->rdpdr->client, GUAC_LOG_ERROR,
|
2013-07-24 22:52:46 +00:00
|
|
|
"Unknown printer I/O request function: 0x%x/0x%x",
|
|
|
|
major_func, minor_func);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static void guac_rdpdr_device_printer_free_handler(guac_rdpdr_device* device) {
|
2016-06-09 20:57:22 +00:00
|
|
|
|
2017-02-14 07:08:54 +00:00
|
|
|
/* Terminate and free print job if open */
|
2017-02-13 22:44:06 +00:00
|
|
|
guac_rdpdr_print_job* job = (guac_rdpdr_print_job*) device->data;
|
2017-02-14 07:08:54 +00:00
|
|
|
if (job != NULL) {
|
|
|
|
guac_rdpdr_print_job_kill(job);
|
2017-02-13 22:44:06 +00:00
|
|
|
guac_rdpdr_print_job_free(job);
|
2017-02-14 07:08:54 +00:00
|
|
|
}
|
2016-06-09 20:57:22 +00:00
|
|
|
|
2013-07-24 22:52:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void guac_rdpdr_register_printer(guac_rdpdrPlugin* rdpdr) {
|
|
|
|
|
2013-07-25 05:20:29 +00:00
|
|
|
int id = rdpdr->devices_registered++;
|
|
|
|
|
2013-07-24 22:52:46 +00:00
|
|
|
/* Get new device */
|
2013-07-25 05:20:29 +00:00
|
|
|
guac_rdpdr_device* device = &(rdpdr->devices[id]);
|
2013-07-24 22:52:46 +00:00
|
|
|
|
|
|
|
/* Init device */
|
2013-07-25 05:20:29 +00:00
|
|
|
device->rdpdr = rdpdr;
|
|
|
|
device->device_id = id;
|
|
|
|
device->device_name = "Guacamole Printer";
|
|
|
|
|
|
|
|
/* Set handlers */
|
2013-07-25 01:28:20 +00:00
|
|
|
device->announce_handler = guac_rdpdr_device_printer_announce_handler;
|
2013-07-24 22:52:46 +00:00
|
|
|
device->iorequest_handler = guac_rdpdr_device_printer_iorequest_handler;
|
|
|
|
device->free_handler = guac_rdpdr_device_printer_free_handler;
|
|
|
|
|
2017-02-13 22:44:06 +00:00
|
|
|
/* No active print job yet */
|
|
|
|
device->data = NULL;
|
2013-06-23 00:43:05 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|