mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-10 01:48:00 +00:00
TempSeat: fix output interleaving.
Working on the previous commit, I suddenly realised I'd made a mistake in the design of TempSeat: you can't buffer standard output and standard error separately and then replay them one after another, because the interleaving of the two kinds of output might also be significant. (Especially if the consuming Seat doesn't separate them.) Now TempSeat has a single bufchain for all the data, paralleled by a linked list describing each contiguous chunk of it consisting of a single output type. So we can replay the data with both the correct separation _and_ the correct order.
This commit is contained in:
parent
ac47e550c6
commit
71cb9ca487
@ -14,10 +14,28 @@
|
|||||||
|
|
||||||
#include "putty.h"
|
#include "putty.h"
|
||||||
|
|
||||||
|
struct output_chunk {
|
||||||
|
struct output_chunk *next;
|
||||||
|
SeatOutputType type;
|
||||||
|
size_t size;
|
||||||
|
};
|
||||||
|
|
||||||
typedef struct TempSeat TempSeat;
|
typedef struct TempSeat TempSeat;
|
||||||
struct TempSeat {
|
struct TempSeat {
|
||||||
Seat *realseat;
|
Seat *realseat;
|
||||||
bufchain outputs[3]; /* stdout, stderr, auth banner (just in case) */
|
|
||||||
|
/*
|
||||||
|
* Single bufchain to hold all the buffered output, regardless of
|
||||||
|
* its type.
|
||||||
|
*/
|
||||||
|
bufchain output;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* List of pieces of that bufchain that are intended for one or
|
||||||
|
* another output destination
|
||||||
|
*/
|
||||||
|
struct output_chunk *outchunk_head, *outchunk_tail;
|
||||||
|
|
||||||
bool seen_session_started;
|
bool seen_session_started;
|
||||||
bool seen_remote_exit;
|
bool seen_remote_exit;
|
||||||
bool seen_remote_disconnect;
|
bool seen_remote_disconnect;
|
||||||
@ -38,14 +56,24 @@ static size_t tempseat_output(Seat *seat, SeatOutputType type,
|
|||||||
{
|
{
|
||||||
TempSeat *ts = container_of(seat, TempSeat, seat);
|
TempSeat *ts = container_of(seat, TempSeat, seat);
|
||||||
|
|
||||||
size_t index = (size_t)type;
|
bufchain_add(&ts->output, data, len);
|
||||||
assert(index < lenof(ts->outputs));
|
|
||||||
bufchain_add(&ts->outputs[index], data, len);
|
|
||||||
|
|
||||||
size_t total_size = 0;
|
if (!(ts->outchunk_tail && ts->outchunk_tail->type == type)) {
|
||||||
for (size_t i = 0; i < lenof(ts->outputs); i++)
|
struct output_chunk *new_chunk = snew(struct output_chunk);
|
||||||
total_size += bufchain_size(&ts->outputs[i]);
|
|
||||||
return total_size;
|
new_chunk->type = type;
|
||||||
|
new_chunk->size = 0;
|
||||||
|
|
||||||
|
new_chunk->next = NULL;
|
||||||
|
if (ts->outchunk_tail)
|
||||||
|
ts->outchunk_tail->next = new_chunk;
|
||||||
|
else
|
||||||
|
ts->outchunk_head = new_chunk;
|
||||||
|
ts->outchunk_tail = new_chunk;
|
||||||
|
}
|
||||||
|
ts->outchunk_tail->type += len;
|
||||||
|
|
||||||
|
return bufchain_size(&ts->output);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tempseat_notify_session_started(Seat *seat)
|
static void tempseat_notify_session_started(Seat *seat)
|
||||||
@ -302,8 +330,8 @@ Seat *tempseat_new(Seat *realseat)
|
|||||||
ts->seat.vt = &tempseat_vt;
|
ts->seat.vt = &tempseat_vt;
|
||||||
|
|
||||||
ts->realseat = realseat;
|
ts->realseat = realseat;
|
||||||
for (size_t i = 0; i < lenof(ts->outputs); i++)
|
bufchain_init(&ts->output);
|
||||||
bufchain_init(&ts->outputs[i]);
|
ts->outchunk_head = ts->outchunk_tail = NULL;
|
||||||
|
|
||||||
return &ts->seat;
|
return &ts->seat;
|
||||||
}
|
}
|
||||||
@ -324,8 +352,12 @@ void tempseat_free(Seat *seat)
|
|||||||
{
|
{
|
||||||
assert(seat->vt == &tempseat_vt);
|
assert(seat->vt == &tempseat_vt);
|
||||||
TempSeat *ts = container_of(seat, TempSeat, seat);
|
TempSeat *ts = container_of(seat, TempSeat, seat);
|
||||||
for (unsigned i = 0; i < 2; i++)
|
bufchain_clear(&ts->output);
|
||||||
bufchain_clear(&ts->outputs[i]);
|
while (ts->outchunk_head) {
|
||||||
|
struct output_chunk *chunk = ts->outchunk_head;
|
||||||
|
ts->outchunk_head = chunk->next;
|
||||||
|
sfree(chunk);
|
||||||
|
}
|
||||||
sfree(ts);
|
sfree(ts);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -334,15 +366,29 @@ void tempseat_flush(Seat *seat)
|
|||||||
assert(seat->vt == &tempseat_vt);
|
assert(seat->vt == &tempseat_vt);
|
||||||
TempSeat *ts = container_of(seat, TempSeat, seat);
|
TempSeat *ts = container_of(seat, TempSeat, seat);
|
||||||
|
|
||||||
/* Empty the output bufchains into the real seat */
|
/* Empty the output bufchains into the real seat, taking care to
|
||||||
for (size_t i = 0; i < lenof(ts->outputs); i++) {
|
* preserve both separation and interleaving */
|
||||||
while (bufchain_size(&ts->outputs[i])) {
|
while (bufchain_size(&ts->output)) {
|
||||||
ptrlen pl = bufchain_prefix(&ts->outputs[i]);
|
ptrlen pl = bufchain_prefix(&ts->output);
|
||||||
seat_output(ts->realseat, i, pl.ptr, pl.len);
|
|
||||||
bufchain_consume(&ts->outputs[i], pl.len);
|
assert(ts->outchunk_head);
|
||||||
|
struct output_chunk *chunk = ts->outchunk_head;
|
||||||
|
|
||||||
|
if (pl.len > chunk->size)
|
||||||
|
pl.len = chunk->size;
|
||||||
|
|
||||||
|
seat_output(ts->realseat, chunk->type, pl.ptr, pl.len);
|
||||||
|
bufchain_consume(&ts->output, pl.len);
|
||||||
|
chunk->size -= pl.len;
|
||||||
|
if (chunk->size == 0) {
|
||||||
|
ts->outchunk_head = chunk->next;
|
||||||
|
sfree(chunk);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* That should have exactly emptied the output chunk list too */
|
||||||
|
assert(!ts->outchunk_head);
|
||||||
|
|
||||||
/* Pass on any other kinds of event we've buffered */
|
/* Pass on any other kinds of event we've buffered */
|
||||||
if (ts->seen_session_started)
|
if (ts->seen_session_started)
|
||||||
seat_notify_session_started(ts->realseat);
|
seat_notify_session_started(ts->realseat);
|
||||||
|
Loading…
Reference in New Issue
Block a user