/* * "Software pw3270, desenvolvido com base nos códigos fontes do WC3270 e X3270 * (Paul Mattes Paul.Mattes@usa.net), de emulação de terminal 3270 para acesso a * aplicativos mainframe. Registro no INPI sob o nome G3270. * * Copyright (C) <2008> * * Este programa é software livre. Você pode redistribuí-lo e/ou modificá-lo sob * os termos da GPL v.2 - Licença Pública Geral GNU, conforme publicado pela * Free Software Foundation. * * Este programa é distribuído na expectativa de ser útil, mas SEM QUALQUER * GARANTIA; sem mesmo a garantia implícita de COMERCIALIZAÇÃO ou de ADEQUAÇÃO * A QUALQUER PROPÓSITO EM PARTICULAR. Consulte a Licença Pública Geral GNU para * obter mais detalhes. * * Você deve ter recebido uma cópia da Licença Pública Geral GNU junto com este * programa; se não, escreva para a Free Software Foundation, Inc., 51 Franklin * St, Fifth Floor, Boston, MA 02110-1301 USA * * Este programa está nomeado como - e possui - linhas de código. * * Contatos: * * perry.werneck@gmail.com (Alexandre Perry de Souza Werneck) * erico.mendonca@gmail.com (Erico Mascarenhas Mendonça) * */ /** * @file src/lib3270++/windows/request.cc * * @brief Implements WIN32 Pipe Based IPC. * * @author perry.werneck@gmail.com * */ #include "../private.h" using std::string; /*---[ Implement ]----------------------------------------------------------------------------------*/ namespace TN3270 { #define PIPE_BUFFER_LENGTH 8192 IPC::Request::Request(const Session &session) { this->hPipe = session.hPipe; in.length = PIPE_BUFFER_LENGTH; in.used = 0; in.block = new uint8_t[in.length]; out.length = PIPE_BUFFER_LENGTH; out.used = 0; out.block = new uint8_t[out.length]; } IPC::Request::Request(const Session &session, const char *method) : Request(session) { // Add name strcpy((char *) out.block, method); out.used += strlen((char *) method) + 1; // Add ID *((uint16_t *) (out.block + out.used)) = (uint16_t) 3; out.used += sizeof(uint16_t); } IPC::Request::Request(const Session &session, bool isSet, const char *property) : Request(session) { // Add name strcpy((char *) out.block, property); out.used += strlen((char *) property) + 1; // Add ID (SetProperty = 2, getProperty = 1) *((uint16_t *) (out.block + out.used)) = (uint16_t) (isSet ? 2 : 1); out.used += sizeof(uint16_t); } IPC::Request::~Request() { delete[] ((uint8_t *) in.block); delete[] ((uint8_t *) out.block); } /// @brief Store value on data block. IPC::Request::DataBlock * IPC::Request::pushBlock(const void *ptr, size_t length) { if((out.used + length + sizeof(IPC::Request::DataBlock)) >= out.length) { throw std::runtime_error("Too big"); } IPC::Request::DataBlock * rc = (IPC::Request::DataBlock *) (out.block + out.used); memcpy(((uint8_t *) (rc+1)), ((uint8_t *) ptr), length); out.used += (sizeof(IPC::Request::DataBlock) + length); return rc; } /// @brief Get next argument. IPC::Request::DataBlock * IPC::Request::getNextBlock() const { if((in.current + sizeof(IPC::Request::DataBlock)) >= in.used) { throw std::runtime_error("Out of range"); } return (IPC::Request::DataBlock *) (in.block + in.current); } IPC::Request & IPC::Request::push(const char *arg) { pushBlock(arg, strlen(arg)+1)->type = IPC::Request::String; return *this; } IPC::Request & IPC::Request::push(const bool arg) { uint8_t value = (uint8_t) (arg ? 0xff : 0); pushBlock(&value, sizeof(value))->type = IPC::Request::Boolean; return *this; } IPC::Request & IPC::Request::push(const uint8_t arg) { pushBlock(&arg, sizeof(arg))->type = IPC::Request::Uchar; return *this; } IPC::Request & IPC::Request::push(const int32_t arg) { pushBlock(&arg, sizeof(arg))->type = IPC::Request::Int32; return *this; } IPC::Request & IPC::Request::push(const uint32_t arg) { pushBlock(&arg, sizeof(arg))->type = IPC::Request::Uint32; return *this; } IPC::Request & IPC::Request::pop(std::string &value) { DataBlock * block = getNextBlock(); if(block->type != IPC::Request::String) throw std::runtime_error("Invalid format"); const char *ptr = (const char *) (block+1); in.current += (strlen(ptr)+1+sizeof(DataBlock)); value.assign(ptr); return *this; } IPC::Request & IPC::Request::Request::pop(int &value) { DataBlock * block = getNextBlock(); switch(block->type) { case IPC::Request::Int16: value = * ((int16_t *) (block+1)); in.current += sizeof(int16_t) + sizeof(DataBlock); break; case IPC::Request::Int32: value = * ((int32_t *) (block+1)); in.current += sizeof(int32_t) + sizeof(DataBlock); break; case IPC::Request::Int64: value = * ((int64_t *) (block+1)); in.current += sizeof(int64_t) + sizeof(DataBlock); break; default: throw std::runtime_error("Invalid format"); } return *this; } }