/*! \file pibasetransfer.h
* \ingroup IO-Utils
* \~\brief
* \~english Base class for reliable packet sessions with acknowledgements, pause and resume
* \~russian Базовый класс для надежных пакетных сессий с подтверждениями, паузой и продолжением
*/
/*
PIP - Platform Independent Primitives
Base class for reliable send and receive data in fixed packets with error correction, pause and resume
Andrey Bychkov work.a.b@yandex.ru
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser 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 Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with this program. If not, see .
*/
#ifndef PIBASETRANSFER_H
#define PIBASETRANSFER_H
#include "picrc.h"
#include "pidiagnostics.h"
//! \ingroup IO-Utils
//! \~\brief
//! \~english Base transport for reliable fixed-size packet exchange over an external channel.
//! \~russian Базовый транспорт для надежного обмена пакетами фиксированного размера через внешний канал.
class PIP_EXPORT PIBaseTransfer: public PIObject {
PIOBJECT_SUBCLASS(PIBaseTransfer, PIObject);
public:
//! \~english Constructs transfer with default packet size, timeout and diagnostics.
//! \~russian Создает передачу с размером пакета, таймаутом и диагностикой по умолчанию.
PIBaseTransfer();
//! \~english Stops active transfer state and owned diagnostics.
//! \~russian Останавливает активное состояние передачи и встроенную диагностику.
~PIBaseTransfer();
#pragma pack(push, 1)
//! \~english Common header placed before every protocol packet.
//! \~russian Общий заголовок, размещаемый перед каждым пакетом протокола.
struct PIP_EXPORT PacketHeader {
//! \~english Packet signature used to recognize transfer packets.
//! \~russian Сигнатура пакета, используемая для распознавания пакетов передачи.
uint sig;
//! \~english Packet type from the transfer protocol.
//! \~russian Тип пакета из протокола передачи.
int type; // PacketType
//! \~english Session identifier for current transfer exchange.
//! \~russian Идентификатор сессии текущего обмена.
int session_id;
//! \~english Sequential packet identifier inside the session.
//! \~russian Последовательный идентификатор пакета внутри сессии.
uint id;
//! \~english CRC calculated for the packet payload.
//! \~russian CRC, вычисленная для полезной нагрузки пакета.
uint crc;
//! \~english Returns whether the packet signature matches \a packetSignature().
//! \~russian Возвращает, совпадает ли сигнатура пакета с \a packetSignature().
bool check_sig() { return (sig == signature); }
};
//! \~english Logical fragment description used to split data into packets.
//! \~russian Описание логического фрагмента, используемое для разбиения данных на пакеты.
struct PIP_EXPORT Part {
//! \~english Constructs part metadata for item "id_", fragment size "size_" and offset "start_".
//! \~russian Создает метаданные части для элемента "id_", размера фрагмента "size_" и смещения "start_".
Part(uint id_ = 0, ullong size_ = 0, ullong start_ = 0): id(id_), size(size_), start(start_) {}
//! \~english Identifier of the logical item being transferred.
//! \~russian Идентификатор логического элемента, который передается.
uint id;
//! \~english Size of this fragment in bytes.
//! \~russian Размер этого фрагмента в байтах.
ullong size;
//! \~english Byte offset of this fragment inside the logical item.
//! \~russian Смещение этого фрагмента в байтах внутри логического элемента.
ullong start;
};
#pragma pack(pop)
//! \~english Requests cancellation of the current send session.
//! \~russian Запрашивает отмену текущей сессии отправки.
void stopSend();
//! \~english Requests cancellation of the current receive session.
//! \~russian Запрашивает отмену текущей сессии приема.
void stopReceive();
//! \~english Returns whether a send session is active.
//! \~russian Возвращает, активна ли сессия отправки.
bool isSending() const { return is_sending; }
//! \~english Returns whether a receive session is active.
//! \~russian Возвращает, активна ли сессия приема.
bool isReceiving() const { return is_receiving; }
//! \~english Returns whether the transfer is currently paused.
//! \~russian Возвращает, находится ли передача сейчас на паузе.
bool isPause() const { return is_pause; }
//! \~english Switches current session between paused and resumed states.
//! \~russian Переключает текущую сессию между состояниями паузы и продолжения.
void setPause(bool pause_);
//! \~english Sets maximum encoded packet size.
//! \~russian Устанавливает максимальный размер кодированного пакета.
void setPacketSize(int size) { packet_size = size; }
//! \~english Returns maximum encoded packet size.
//! \~russian Возвращает максимальный размер кодированного пакета.
int packetSize() const { return packet_size; }
//! \~english Sets session timeout for start negotiation, acknowledgements and pause recovery.
//! \~russian Устанавливает таймаут сессии для согласования старта, подтверждений и восстановления после паузы.
void setTimeout(double sec);
//! \~english Returns session timeout in seconds.
//! \~russian Возвращает таймаут сессии в секундах.
double timeout() const { return timeout_; }
//! \~english Enables or disables CRC validation for data packets.
//! \~russian Включает или выключает проверку CRC для пакетов данных.
void setCRCEnabled(bool en = true) { crc_enabled = en; }
//! \~english Returns whether CRC validation is enabled.
//! \~russian Возвращает, включена ли проверка CRC.
bool isCRCEnabled() const { return crc_enabled; }
//! \~english Returns short textual state of the current session.
//! \~russian Возвращает краткое текстовое состояние текущей сессии.
PIString stateString() const { return state_string; }
//! \~english Returns acknowledgement map for the current session.
//! \~russian Возвращает карту подтверждений для текущей сессии.
PIString packetMap() const { return pm_string; }
//! \~english Returns total number of bytes planned for the current session.
//! \~russian Возвращает общее число байтов, запланированных для текущей сессии.
llong bytesAll() const { return bytes_all; }
//! \~english Returns number of bytes already processed in the current session.
//! \~russian Возвращает число байтов, уже обработанных в текущей сессии.
llong bytesCur() const { return bytes_cur; }
//! \~english Returns live transport diagnostics.
//! \~russian Возвращает текущую диагностику транспорта.
const PIDiagnostics & diagnostic() { return diag; }
//! \~english Returns the packet signature constant used by the protocol.
//! \~russian Возвращает константу сигнатуры пакета, используемую протоколом.
static uint packetSignature() { return signature; }
EVENT_HANDLER1(void, received, PIByteArray, data);
EVENT_HANDLER(void, stop) {
stopSend();
stopReceive();
}
EVENT_HANDLER(void, pause) { setPause(true); }
EVENT_HANDLER(void, resume) { setPause(false); }
EVENT(receiveStarted);
EVENT(paused);
EVENT(resumed);
EVENT1(receiveFinished, bool, ok);
EVENT(sendStarted);
EVENT1(sendFinished, bool, ok);
EVENT1(sendRequest, PIByteArray &, data);
//! \handlers
//! \{
//!
//! \fn void received(PIByteArray data)
//! \~english Feeds one encoded packet received from the external transport.
//! \~russian Передает один закодированный пакет, полученный от внешнего транспорта.
//!
//! \fn void stop()
//! \~english Stops both sending and receiving sides of the current session.
//! \~russian Останавливает и отправку, и прием текущей сессии.
//!
//! \fn void pause()
//! \~english Switches current session to paused state.
//! \~russian Переводит текущую сессию в состояние паузы.
//!
//! \fn void resume()
//! \~english Resumes the current paused session.
//! \~russian Продолжает текущую приостановленную сессию.
//!
//! \}
//! \events
//! \{
//!
//! \fn void receiveStarted()
//! \~english Emitted when a receive session is accepted and initialized.
//! \~russian Генерируется, когда сессия приема принята и инициализирована.
//!
//! \fn void paused()
//! \~english Emitted when the transfer enters paused state.
//! \~russian Генерируется, когда передача переходит в состояние паузы.
//!
//! \fn void resumed()
//! \~english Emitted when the transfer leaves paused state.
//! \~russian Генерируется, когда передача выходит из состояния паузы.
//!
//! \fn void receiveFinished(bool ok)
//! \~english Emitted when the receive session finishes with result "ok".
//! \~russian Генерируется, когда сессия приема завершается с результатом "ok".
//!
//! \fn void sendStarted()
//! \~english Emitted when a prepared send session starts.
//! \~russian Генерируется при запуске подготовленной сессии отправки.
//!
//! \fn void sendFinished(bool ok)
//! \~english Emitted when the send session finishes with result "ok".
//! \~russian Генерируется, когда сессия отправки завершается с результатом "ok".
//!
//! \fn void sendRequest(PIByteArray data)
//! \~english Emitted for every encoded packet that must be written to the external transport.
//! \~russian Генерируется для каждого закодированного пакета, который нужно записать во внешний транспорт.
//!
//! \}
protected:
//! \~english Builds session packet layout for logical parts in "parts".
//! \~russian Формирует раскладку пакетов сессии для логических частей из "parts".
void buildSession(PIVector parts);
//! \~english Returns payload bytes for one requested logical fragment.
//! \~russian Возвращает байты полезной нагрузки для одного запрошенного логического фрагмента.
virtual PIByteArray buildPacket(Part fi) = 0;
//! \~english Consumes one received logical fragment and optional custom packet header.
//! \~russian Обрабатывает один принятый логический фрагмент и необязательный пользовательский заголовок пакета.
virtual void receivePart(Part fi, PIByteArray ba, PIByteArray pheader) = 0;
//! \~english Called after a new receive session is accepted and initialized.
//! \~russian Вызывается после принятия и инициализации новой сессии приема.
virtual void beginReceive() { ; }
//! \~english Returns custom header bytes inserted after \a PacketHeader in each data packet.
//! \~russian Возвращает байты пользовательского заголовка, вставляемые после \a PacketHeader в каждом пакете данных.
virtual PIByteArray customHeader() { return PIByteArray(); }
//! \~english Runs the prepared send session until success, failure or cancellation.
//! \~russian Выполняет подготовленную сессию отправки до успеха, ошибки или отмены.
bool send_process();
uint packet_header_size, part_header_size;
bool break_, is_sending, is_receiving, is_pause;
PIString state_string;
llong bytes_all, bytes_cur;
private:
enum PacketType {
pt_Unknown,
pt_Data,
pt_ReplySuccess,
pt_ReplyInvalid,
pt_Break,
pt_Start,
pt_Pause
};
#pragma pack(push, 1)
struct PIP_EXPORT StartRequest {
uint packets;
ullong size;
};
#pragma pack(pop)
BINARY_STREAM_FRIEND(PIBaseTransfer::StartRequest);
void processData(int id, PIByteArray & data);
PIByteArray build_packet(int id);
int checkSession();
void sendBreak(int session_id);
void sendReply(PacketType reply);
bool getStartRequest();
bool finish_send(bool ok);
void finish_receive(bool ok, bool quet = false);
static const uint signature;
int packet_size, packets_count;
double timeout_;
PIVector> session;
PIVector replies;
PITimeMeasurer send_tm, pause_tm;
PacketHeader header;
CRC_16 crc;
int send_queue;
int send_up;
PIDiagnostics diag;
PIMutex mutex_session;
PIMutex mutex_send;
PIMutex mutex_header;
bool crc_enabled;
PIString pm_string;
};
BINARY_STREAM_WRITE(PIBaseTransfer::PacketHeader) {
s << v.sig << v.type << v.session_id << v.id << v.crc;
return s;
}
BINARY_STREAM_READ(PIBaseTransfer::PacketHeader) {
s >> v.sig >> v.type >> v.session_id >> v.id >> v.crc;
return s;
}
BINARY_STREAM_WRITE(PIBaseTransfer::Part) {
s << v.id << v.size << v.start;
return s;
}
BINARY_STREAM_READ(PIBaseTransfer::Part) {
s >> v.id >> v.size >> v.start;
return s;
}
BINARY_STREAM_WRITE(PIBaseTransfer::StartRequest) {
s << v.packets << v.size;
return s;
}
BINARY_STREAM_READ(PIBaseTransfer::StartRequest) {
s >> v.packets >> v.size;
return s;
}
inline PICout operator<<(PICout s, const PIBaseTransfer::Part & v) {
s.saveAndSetControls(0);
s << "Part(\"" << v.id << "\", " << PIString::readableSize(v.start) << " b | " << PIString::readableSize(v.size) << " b)";
s.restoreControls();
return s;
}
#endif // PIBASETRANSFER_H