mirror of
https://gitlab.uni-freiburg.de/opensourcevdi/spice
synced 2025-12-26 22:48:19 +00:00
Now the push is done automatically when a PipeItem is added
(cfr commit 5c460de1a3
"worker: push data when clients can receive them"),
forcing a push cause only network fragmentation and is required only if
you are handling data in a polling loop (and thus, you are preventing
the default event loop from running).
Signed-off-by: Frediano Ziglio <fziglio@redhat.com>
Acked-by: Christophe Fergeau <cfergeau@redhat.com>
200 lines
8.3 KiB
C
200 lines
8.3 KiB
C
/*
|
|
Copyright (C) 2009-2016 Red Hat, Inc.
|
|
|
|
This library 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 2.1 of the License, or (at your option) any later version.
|
|
|
|
This library 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 library; if not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#ifndef RED_CHANNEL_CLIENT_H_
|
|
#define RED_CHANNEL_CLIENT_H_
|
|
|
|
#include <glib-object.h>
|
|
#include <gio/gio.h>
|
|
#include <common/marshaller.h>
|
|
|
|
#include "red-pipe-item.h"
|
|
#include "reds-stream.h"
|
|
#include "red-channel.h"
|
|
|
|
G_BEGIN_DECLS
|
|
|
|
#define RED_TYPE_CHANNEL_CLIENT red_channel_client_get_type()
|
|
|
|
#define RED_CHANNEL_CLIENT(obj) \
|
|
(G_TYPE_CHECK_INSTANCE_CAST((obj), RED_TYPE_CHANNEL_CLIENT, RedChannelClient))
|
|
#define RED_CHANNEL_CLIENT_CLASS(klass) \
|
|
(G_TYPE_CHECK_CLASS_CAST((klass), RED_TYPE_CHANNEL_CLIENT, RedChannelClientClass))
|
|
#define RED_IS_CHANNEL_CLIENT(obj) \
|
|
(G_TYPE_CHECK_INSTANCE_TYPE((obj), RED_TYPE_CHANNEL_CLIENT))
|
|
#define RED_IS_CHANNEL_CLIENT_CLASS(klass) \
|
|
(G_TYPE_CHECK_CLASS_TYPE((klass), RED_TYPE_CHANNEL_CLIENT))
|
|
#define RED_CHANNEL_CLIENT_GET_CLASS(obj) \
|
|
(G_TYPE_INSTANCE_GET_CLASS((obj), RED_TYPE_CHANNEL_CLIENT, RedChannelClientClass))
|
|
|
|
typedef struct RedChannelClient RedChannelClient;
|
|
typedef struct RedChannelClientClass RedChannelClientClass;
|
|
typedef struct RedChannelClientPrivate RedChannelClientPrivate;
|
|
|
|
GType red_channel_client_get_type(void) G_GNUC_CONST;
|
|
|
|
gboolean red_channel_client_is_connected(RedChannelClient *rcc);
|
|
void red_channel_client_default_migrate(RedChannelClient *rcc);
|
|
bool red_channel_client_is_waiting_for_migrate_data(RedChannelClient *rcc);
|
|
void red_channel_client_destroy(RedChannelClient *rcc);
|
|
bool red_channel_client_test_remote_common_cap(RedChannelClient *rcc, uint32_t cap);
|
|
bool red_channel_client_test_remote_cap(RedChannelClient *rcc, uint32_t cap);
|
|
/* shutdown is the only safe thing to do out of the client/channel
|
|
* thread. It will not touch the rings, just shutdown the socket.
|
|
* It should be followed by some way to guarantee a disconnection. */
|
|
void red_channel_client_shutdown(RedChannelClient *rcc);
|
|
/* handles general channel msgs from the client */
|
|
bool red_channel_client_handle_message(RedChannelClient *rcc, uint16_t type,
|
|
uint32_t size, void *message);
|
|
/* when preparing send_data: should call init and then use marshaller */
|
|
void red_channel_client_init_send_data(RedChannelClient *rcc, uint16_t msg_type);
|
|
|
|
uint64_t red_channel_client_get_message_serial(RedChannelClient *channel);
|
|
|
|
/* When sending a msg. Should first call red_channel_client_begin_send_message.
|
|
* It will first send the pending urgent data, if there is any, and then
|
|
* the rest of the data.
|
|
*/
|
|
void red_channel_client_begin_send_message(RedChannelClient *rcc);
|
|
|
|
/*
|
|
* Stores the current send data, and switches to urgent send data.
|
|
* When it begins the actual send, it will send first the urgent data
|
|
* and afterward the rest of the data.
|
|
* Should be called only if during the marshalling of on message,
|
|
* the need to send another message, before, rises.
|
|
* Important: the serial of the non-urgent sent data, will be succeeded.
|
|
* return: the urgent send data marshaller
|
|
*/
|
|
SpiceMarshaller *red_channel_client_switch_to_urgent_sender(RedChannelClient *rcc);
|
|
|
|
/* returns -1 if we don't have an estimation */
|
|
int red_channel_client_get_roundtrip_ms(RedChannelClient *rcc);
|
|
|
|
/* Checks periodically if the connection is still alive */
|
|
void red_channel_client_start_connectivity_monitoring(RedChannelClient *rcc, uint32_t timeout_ms);
|
|
|
|
void red_channel_client_pipe_add_push(RedChannelClient *rcc, RedPipeItem *item);
|
|
void red_channel_client_pipe_add(RedChannelClient *rcc, RedPipeItem *item);
|
|
void red_channel_client_pipe_add_after(RedChannelClient *rcc, RedPipeItem *item, RedPipeItem *pos);
|
|
void red_channel_client_pipe_add_after_pos(RedChannelClient *rcc, RedPipeItem *item, GList *pos);
|
|
int red_channel_client_pipe_item_is_linked(RedChannelClient *rcc, RedPipeItem *item);
|
|
void red_channel_client_pipe_remove_and_release(RedChannelClient *rcc, RedPipeItem *item);
|
|
void red_channel_client_pipe_remove_and_release_pos(RedChannelClient *rcc, GList *item_pos);
|
|
void red_channel_client_pipe_add_tail(RedChannelClient *rcc, RedPipeItem *item);
|
|
/* for types that use this routine -> the pipe item should be freed */
|
|
void red_channel_client_pipe_add_type(RedChannelClient *rcc, int pipe_item_type);
|
|
RedPipeItem *red_channel_client_new_empty_msg(int msg_type);
|
|
void red_channel_client_pipe_add_empty_msg(RedChannelClient *rcc, int msg_type);
|
|
gboolean red_channel_client_pipe_is_empty(RedChannelClient *rcc);
|
|
uint32_t red_channel_client_get_pipe_size(RedChannelClient *rcc);
|
|
GQueue* red_channel_client_get_pipe(RedChannelClient *rcc);
|
|
bool red_channel_client_is_mini_header(RedChannelClient *rcc);
|
|
|
|
void red_channel_client_ack_zero_messages_window(RedChannelClient *rcc);
|
|
void red_channel_client_ack_set_client_window(RedChannelClient *rcc, int client_window);
|
|
void red_channel_client_push_set_ack(RedChannelClient *rcc);
|
|
|
|
gboolean red_channel_client_is_blocked(RedChannelClient *rcc);
|
|
|
|
/* helper for channels that have complex logic that can possibly ready a send */
|
|
int red_channel_client_send_message_pending(RedChannelClient *rcc);
|
|
|
|
gboolean red_channel_client_no_item_being_sent(RedChannelClient *rcc);
|
|
void red_channel_client_push(RedChannelClient *rcc);
|
|
void red_channel_client_receive(RedChannelClient *rcc);
|
|
void red_channel_client_send(RedChannelClient *rcc);
|
|
void red_channel_client_disconnect(RedChannelClient *rcc);
|
|
|
|
/* Note: the valid times to call red_channel_get_marshaller are just during send_item callback. */
|
|
SpiceMarshaller *red_channel_client_get_marshaller(RedChannelClient *rcc);
|
|
RedsStream *red_channel_client_get_stream(RedChannelClient *rcc);
|
|
RedClient *red_channel_client_get_client(RedChannelClient *rcc);
|
|
|
|
/* Note that the header is valid only between red_channel_reset_send_data and
|
|
* red_channel_begin_send_message.*/
|
|
void red_channel_client_set_header_sub_list(RedChannelClient *rcc, uint32_t sub_list);
|
|
|
|
/*
|
|
* blocking functions.
|
|
*
|
|
* timeout is in nano sec. -1 for no timeout.
|
|
*
|
|
* Return: TRUE if waiting succeeded. FALSE if timeout expired.
|
|
*/
|
|
|
|
bool red_channel_client_wait_pipe_item_sent(RedChannelClient *rcc,
|
|
GList *item_pos,
|
|
int64_t timeout);
|
|
bool red_channel_client_wait_outgoing_item(RedChannelClient *rcc,
|
|
int64_t timeout);
|
|
|
|
RedChannel* red_channel_client_get_channel(RedChannelClient *rcc);
|
|
|
|
void red_channel_client_semi_seamless_migration_complete(RedChannelClient *rcc);
|
|
void red_channel_client_init_outgoing_messages_window(RedChannelClient *rcc);
|
|
|
|
gboolean red_channel_client_set_migration_seamless(RedChannelClient *rcc);
|
|
void red_channel_client_set_destroying(RedChannelClient *rcc);
|
|
bool red_channel_client_is_destroying(RedChannelClient *rcc);
|
|
|
|
struct RedChannelClient
|
|
{
|
|
GObject parent;
|
|
|
|
RedChannelClientPrivate *priv;
|
|
};
|
|
|
|
struct RedChannelClientClass
|
|
{
|
|
GObjectClass parent_class;
|
|
|
|
/* configure socket connected to the client */
|
|
bool (*config_socket)(RedChannelClient *rcc);
|
|
uint8_t *(*alloc_recv_buf)(RedChannelClient *channel, uint16_t type, uint32_t size);
|
|
void (*release_recv_buf)(RedChannelClient *channel, uint16_t type, uint32_t size, uint8_t *msg);
|
|
|
|
void (*on_disconnect)(RedChannelClient *rcc);
|
|
};
|
|
|
|
#define SPICE_SERVER_ERROR spice_server_error_quark()
|
|
GQuark spice_server_error_quark(void);
|
|
|
|
typedef enum
|
|
{
|
|
SPICE_SERVER_ERROR_FAILED
|
|
} SpiceServerError;
|
|
|
|
/* Messages handled by RedChannel
|
|
* SET_ACK - sent to client on channel connection
|
|
* Note that the numbers don't have to correspond to spice message types,
|
|
* but we keep the 100 first allocated for base channel approach.
|
|
* */
|
|
enum {
|
|
RED_PIPE_ITEM_TYPE_SET_ACK=1,
|
|
RED_PIPE_ITEM_TYPE_MIGRATE,
|
|
RED_PIPE_ITEM_TYPE_EMPTY_MSG,
|
|
RED_PIPE_ITEM_TYPE_PING,
|
|
RED_PIPE_ITEM_TYPE_MARKER,
|
|
|
|
RED_PIPE_ITEM_TYPE_CHANNEL_BASE=101,
|
|
};
|
|
|
|
G_END_DECLS
|
|
|
|
#endif /* RED_CHANNEL_CLIENT_H_ */
|