mirror of
https://github.com/tmux/tmux.git
synced 2025-04-20 19:38:48 +00:00
Merge branch 'obsd-master'
This commit is contained in:
commit
bb5a1925ab
51
grid.c
51
grid.c
@ -100,11 +100,11 @@ grid_get_extended_cell(struct grid_line *gl, struct grid_cell_entry *gce,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Set cell as extended. */
|
/* Set cell as extended. */
|
||||||
static struct grid_cell *
|
static struct grid_extd_entry *
|
||||||
grid_extended_cell(struct grid_line *gl, struct grid_cell_entry *gce,
|
grid_extended_cell(struct grid_line *gl, struct grid_cell_entry *gce,
|
||||||
const struct grid_cell *gc)
|
const struct grid_cell *gc)
|
||||||
{
|
{
|
||||||
struct grid_cell *gcp;
|
struct grid_extd_entry *gee;
|
||||||
int flags = (gc->flags & ~GRID_FLAG_CLEARED);
|
int flags = (gc->flags & ~GRID_FLAG_CLEARED);
|
||||||
|
|
||||||
if (~gce->flags & GRID_FLAG_EXTENDED)
|
if (~gce->flags & GRID_FLAG_EXTENDED)
|
||||||
@ -113,10 +113,14 @@ grid_extended_cell(struct grid_line *gl, struct grid_cell_entry *gce,
|
|||||||
fatalx("offset too big");
|
fatalx("offset too big");
|
||||||
gl->flags |= GRID_LINE_EXTENDED;
|
gl->flags |= GRID_LINE_EXTENDED;
|
||||||
|
|
||||||
gcp = &gl->extddata[gce->offset];
|
gee = &gl->extddata[gce->offset];
|
||||||
memcpy(gcp, gc, sizeof *gcp);
|
gee->data = utf8_map_big(&gc->data);
|
||||||
gcp->flags = flags;
|
gee->attr = gc->attr;
|
||||||
return (gcp);
|
gee->flags = flags;
|
||||||
|
gee->fg = gc->fg;
|
||||||
|
gee->bg = gc->bg;
|
||||||
|
gee->us = gc->us;
|
||||||
|
return (gee);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Free up unused extended cells. */
|
/* Free up unused extended cells. */
|
||||||
@ -124,9 +128,9 @@ static void
|
|||||||
grid_compact_line(struct grid_line *gl)
|
grid_compact_line(struct grid_line *gl)
|
||||||
{
|
{
|
||||||
int new_extdsize = 0;
|
int new_extdsize = 0;
|
||||||
struct grid_cell *new_extddata;
|
struct grid_extd_entry *new_extddata;
|
||||||
struct grid_cell_entry *gce;
|
struct grid_cell_entry *gce;
|
||||||
struct grid_cell *gc;
|
struct grid_extd_entry *gee;
|
||||||
u_int px, idx;
|
u_int px, idx;
|
||||||
|
|
||||||
if (gl->extdsize == 0)
|
if (gl->extdsize == 0)
|
||||||
@ -150,8 +154,8 @@ grid_compact_line(struct grid_line *gl)
|
|||||||
for (px = 0; px < gl->cellsize; px++) {
|
for (px = 0; px < gl->cellsize; px++) {
|
||||||
gce = &gl->celldata[px];
|
gce = &gl->celldata[px];
|
||||||
if (gce->flags & GRID_FLAG_EXTENDED) {
|
if (gce->flags & GRID_FLAG_EXTENDED) {
|
||||||
gc = &gl->extddata[gce->offset];
|
gee = &gl->extddata[gce->offset];
|
||||||
memcpy(&new_extddata[idx], gc, sizeof *gc);
|
memcpy(&new_extddata[idx], gee, sizeof *gee);
|
||||||
gce->offset = idx++;
|
gce->offset = idx++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -181,17 +185,14 @@ grid_clear_cell(struct grid *gd, u_int px, u_int py, u_int bg)
|
|||||||
{
|
{
|
||||||
struct grid_line *gl = &gd->linedata[py];
|
struct grid_line *gl = &gd->linedata[py];
|
||||||
struct grid_cell_entry *gce = &gl->celldata[px];
|
struct grid_cell_entry *gce = &gl->celldata[px];
|
||||||
struct grid_cell *gc;
|
struct grid_extd_entry *gee;
|
||||||
|
|
||||||
memcpy(gce, &grid_cleared_entry, sizeof *gce);
|
memcpy(gce, &grid_cleared_entry, sizeof *gce);
|
||||||
if (bg != 8) {
|
if (bg != 8) {
|
||||||
if (bg & COLOUR_FLAG_RGB) {
|
if (bg & COLOUR_FLAG_RGB) {
|
||||||
grid_get_extended_cell(gl, gce, gce->flags);
|
grid_get_extended_cell(gl, gce, gce->flags);
|
||||||
gl->flags |= GRID_LINE_EXTENDED;
|
gee = grid_extended_cell(gl, gce, &grid_cleared_cell);
|
||||||
|
gee->bg = bg;
|
||||||
gc = &gl->extddata[gce->offset];
|
|
||||||
memcpy(gc, &grid_cleared_cell, sizeof *gc);
|
|
||||||
gc->bg = bg;
|
|
||||||
} else {
|
} else {
|
||||||
if (bg & COLOUR_FLAG_256)
|
if (bg & COLOUR_FLAG_256)
|
||||||
gce->flags |= GRID_FLAG_BG256;
|
gce->flags |= GRID_FLAG_BG256;
|
||||||
@ -483,12 +484,20 @@ static void
|
|||||||
grid_get_cell1(struct grid_line *gl, u_int px, struct grid_cell *gc)
|
grid_get_cell1(struct grid_line *gl, u_int px, struct grid_cell *gc)
|
||||||
{
|
{
|
||||||
struct grid_cell_entry *gce = &gl->celldata[px];
|
struct grid_cell_entry *gce = &gl->celldata[px];
|
||||||
|
struct grid_extd_entry *gee;
|
||||||
|
|
||||||
if (gce->flags & GRID_FLAG_EXTENDED) {
|
if (gce->flags & GRID_FLAG_EXTENDED) {
|
||||||
if (gce->offset >= gl->extdsize)
|
if (gce->offset >= gl->extdsize)
|
||||||
memcpy(gc, &grid_default_cell, sizeof *gc);
|
memcpy(gc, &grid_default_cell, sizeof *gc);
|
||||||
else
|
else {
|
||||||
memcpy(gc, &gl->extddata[gce->offset], sizeof *gc);
|
gee = &gl->extddata[gce->offset];
|
||||||
|
gc->flags = gee->flags;
|
||||||
|
gc->attr = gee->attr;
|
||||||
|
gc->fg = gee->fg;
|
||||||
|
gc->bg = gee->bg;
|
||||||
|
gc->us = gee->us;
|
||||||
|
utf8_get_big(gee->data, &gc->data);
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -545,7 +554,7 @@ grid_set_cells(struct grid *gd, u_int px, u_int py, const struct grid_cell *gc,
|
|||||||
{
|
{
|
||||||
struct grid_line *gl;
|
struct grid_line *gl;
|
||||||
struct grid_cell_entry *gce;
|
struct grid_cell_entry *gce;
|
||||||
struct grid_cell *gcp;
|
struct grid_extd_entry *gee;
|
||||||
u_int i;
|
u_int i;
|
||||||
|
|
||||||
if (grid_check_y(gd, __func__, py) != 0)
|
if (grid_check_y(gd, __func__, py) != 0)
|
||||||
@ -560,8 +569,8 @@ grid_set_cells(struct grid *gd, u_int px, u_int py, const struct grid_cell *gc,
|
|||||||
for (i = 0; i < slen; i++) {
|
for (i = 0; i < slen; i++) {
|
||||||
gce = &gl->celldata[px + i];
|
gce = &gl->celldata[px + i];
|
||||||
if (grid_need_extended_cell(gce, gc)) {
|
if (grid_need_extended_cell(gce, gc)) {
|
||||||
gcp = grid_extended_cell(gl, gce, gc);
|
gee = grid_extended_cell(gl, gce, gc);
|
||||||
utf8_set(&gcp->data, s[i]);
|
gee->data = utf8_set_big(s[i], 1);
|
||||||
} else
|
} else
|
||||||
grid_store_cell(gce, gc, s[i]);
|
grid_store_cell(gce, gc, s[i]);
|
||||||
}
|
}
|
||||||
|
31
tmux.h
31
tmux.h
@ -599,11 +599,11 @@ struct msg_write_close {
|
|||||||
#define MOTION_MOUSE_MODES (MODE_MOUSE_BUTTON|MODE_MOUSE_ALL)
|
#define MOTION_MOUSE_MODES (MODE_MOUSE_BUTTON|MODE_MOUSE_ALL)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* A single UTF-8 character. UTF8_SIZE must be big enough to hold
|
* A single UTF-8 character. UTF8_SIZE must be big enough to hold combining
|
||||||
* combining characters as well, currently at most five (of three
|
* characters as well. It can't be more than 32 bytes without changes to how
|
||||||
* bytes) are supported.
|
* big characters are stored.
|
||||||
*/
|
*/
|
||||||
#define UTF8_SIZE 18
|
#define UTF8_SIZE 21
|
||||||
struct utf8_data {
|
struct utf8_data {
|
||||||
u_char data[UTF8_SIZE];
|
u_char data[UTF8_SIZE];
|
||||||
|
|
||||||
@ -611,7 +611,7 @@ struct utf8_data {
|
|||||||
u_char size;
|
u_char size;
|
||||||
|
|
||||||
u_char width; /* 0xff if invalid */
|
u_char width; /* 0xff if invalid */
|
||||||
} __packed;
|
};
|
||||||
enum utf8_state {
|
enum utf8_state {
|
||||||
UTF8_MORE,
|
UTF8_MORE,
|
||||||
UTF8_DONE,
|
UTF8_DONE,
|
||||||
@ -665,13 +665,25 @@ enum utf8_state {
|
|||||||
|
|
||||||
/* Grid cell data. */
|
/* Grid cell data. */
|
||||||
struct grid_cell {
|
struct grid_cell {
|
||||||
struct utf8_data data; /* 21 bytes */
|
struct utf8_data data;
|
||||||
|
u_short attr;
|
||||||
|
u_char flags;
|
||||||
|
int fg;
|
||||||
|
int bg;
|
||||||
|
int us;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Grid extended cell entry. */
|
||||||
|
struct grid_extd_entry {
|
||||||
|
uint32_t data;
|
||||||
u_short attr;
|
u_short attr;
|
||||||
u_char flags;
|
u_char flags;
|
||||||
int fg;
|
int fg;
|
||||||
int bg;
|
int bg;
|
||||||
int us;
|
int us;
|
||||||
} __packed;
|
} __packed;
|
||||||
|
|
||||||
|
/* Grid cell entry. */
|
||||||
struct grid_cell_entry {
|
struct grid_cell_entry {
|
||||||
u_char flags;
|
u_char flags;
|
||||||
union {
|
union {
|
||||||
@ -692,7 +704,7 @@ struct grid_line {
|
|||||||
struct grid_cell_entry *celldata;
|
struct grid_cell_entry *celldata;
|
||||||
|
|
||||||
u_int extdsize;
|
u_int extdsize;
|
||||||
struct grid_cell *extddata;
|
struct grid_extd_entry *extddata;
|
||||||
|
|
||||||
int flags;
|
int flags;
|
||||||
} __packed;
|
} __packed;
|
||||||
@ -2879,6 +2891,9 @@ u_int session_group_attached_count(struct session_group *);
|
|||||||
void session_renumber_windows(struct session *);
|
void session_renumber_windows(struct session *);
|
||||||
|
|
||||||
/* utf8.c */
|
/* utf8.c */
|
||||||
|
uint32_t utf8_set_big(char, u_int);
|
||||||
|
uint32_t utf8_map_big(const struct utf8_data *);
|
||||||
|
void utf8_get_big(uint32_t, struct utf8_data *);
|
||||||
void utf8_set(struct utf8_data *, u_char);
|
void utf8_set(struct utf8_data *, u_char);
|
||||||
void utf8_copy(struct utf8_data *, const struct utf8_data *);
|
void utf8_copy(struct utf8_data *, const struct utf8_data *);
|
||||||
enum utf8_state utf8_open(struct utf8_data *, u_char);
|
enum utf8_state utf8_open(struct utf8_data *, u_char);
|
||||||
|
174
utf8.c
174
utf8.c
@ -28,6 +28,180 @@
|
|||||||
|
|
||||||
static int utf8_width(wchar_t);
|
static int utf8_width(wchar_t);
|
||||||
|
|
||||||
|
struct utf8_big_item {
|
||||||
|
u_int index;
|
||||||
|
RB_ENTRY(utf8_big_item) entry;
|
||||||
|
|
||||||
|
char data[UTF8_SIZE];
|
||||||
|
u_char size;
|
||||||
|
};
|
||||||
|
RB_HEAD(utf8_big_tree, utf8_big_item);
|
||||||
|
|
||||||
|
static int
|
||||||
|
utf8_big_cmp(struct utf8_big_item *bi1, struct utf8_big_item *bi2)
|
||||||
|
{
|
||||||
|
if (bi1->size < bi2->size)
|
||||||
|
return (-1);
|
||||||
|
if (bi1->size > bi2->size)
|
||||||
|
return (1);
|
||||||
|
return (memcmp(bi1->data, bi2->data, bi1->size));
|
||||||
|
}
|
||||||
|
RB_GENERATE_STATIC(utf8_big_tree, utf8_big_item, entry, utf8_big_cmp);
|
||||||
|
static struct utf8_big_tree utf8_big_tree = RB_INITIALIZER(utf8_big_tree);
|
||||||
|
|
||||||
|
static struct utf8_big_item *utf8_big_list;
|
||||||
|
static u_int utf8_big_list_size;
|
||||||
|
static u_int utf8_big_list_used;
|
||||||
|
|
||||||
|
union utf8_big_map {
|
||||||
|
uint32_t value;
|
||||||
|
struct {
|
||||||
|
u_char flags;
|
||||||
|
#define UTF8_BIG_SIZE 0x1f
|
||||||
|
#define UTF8_BIG_WIDTH2 0x20
|
||||||
|
|
||||||
|
u_char data[3];
|
||||||
|
};
|
||||||
|
} __packed;
|
||||||
|
|
||||||
|
static const union utf8_big_map utf8_big_space1 = {
|
||||||
|
.flags = 1,
|
||||||
|
.data = " "
|
||||||
|
};
|
||||||
|
static const union utf8_big_map utf8_big_space2 = {
|
||||||
|
.flags = UTF8_BIG_WIDTH2|2,
|
||||||
|
.data = " "
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Get a big item by index. */
|
||||||
|
static struct utf8_big_item *
|
||||||
|
utf8_get_big_item(const char *data, size_t size)
|
||||||
|
{
|
||||||
|
struct utf8_big_item bi;
|
||||||
|
|
||||||
|
memcpy(bi.data, data, size);
|
||||||
|
bi.size = size;
|
||||||
|
|
||||||
|
return (RB_FIND(utf8_big_tree, &utf8_big_tree, &bi));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Add a big item. */
|
||||||
|
static int
|
||||||
|
utf8_put_big_item(const char *data, size_t size, u_int *index)
|
||||||
|
{
|
||||||
|
struct utf8_big_item *bi;
|
||||||
|
|
||||||
|
bi = utf8_get_big_item(data, size);
|
||||||
|
if (bi != NULL) {
|
||||||
|
*index = bi->index;
|
||||||
|
log_debug("%s: have %.*s at %u", __func__, (int)size, data,
|
||||||
|
*index);
|
||||||
|
return (0);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (utf8_big_list_used == utf8_big_list_size) {
|
||||||
|
if (utf8_big_list_size == 0xffffff)
|
||||||
|
return (-1);
|
||||||
|
if (utf8_big_list_size == 0)
|
||||||
|
utf8_big_list_size = 256;
|
||||||
|
else if (utf8_big_list_size > 0x7fffff)
|
||||||
|
utf8_big_list_size = 0xffffff;
|
||||||
|
else
|
||||||
|
utf8_big_list_size *= 2;
|
||||||
|
utf8_big_list = xreallocarray(utf8_big_list, utf8_big_list_size,
|
||||||
|
sizeof *utf8_big_list);
|
||||||
|
}
|
||||||
|
*index = utf8_big_list_used++;
|
||||||
|
|
||||||
|
bi = &utf8_big_list[*index];
|
||||||
|
bi->index = *index;
|
||||||
|
memcpy(bi->data, data, size);
|
||||||
|
bi->size = size;
|
||||||
|
RB_INSERT(utf8_big_tree, &utf8_big_tree, bi);
|
||||||
|
|
||||||
|
log_debug("%s: added %.*s at %u", __func__, (int)size, data, *index);
|
||||||
|
return (0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get UTF-8 as index into buffer. */
|
||||||
|
uint32_t
|
||||||
|
utf8_map_big(const struct utf8_data *ud)
|
||||||
|
{
|
||||||
|
union utf8_big_map m = { .value = 0 };
|
||||||
|
u_int o;
|
||||||
|
const char *data = ud->data;
|
||||||
|
size_t size = ud->size;
|
||||||
|
|
||||||
|
if (ud->width != 1 && ud->width != 2)
|
||||||
|
return (utf8_big_space1.value);
|
||||||
|
|
||||||
|
if (size > UTF8_BIG_SIZE)
|
||||||
|
goto fail;
|
||||||
|
if (size == 1)
|
||||||
|
return (utf8_set_big(data[0], 1));
|
||||||
|
|
||||||
|
m.flags = size;
|
||||||
|
if (ud->width == 2)
|
||||||
|
m.flags |= UTF8_BIG_WIDTH2;
|
||||||
|
|
||||||
|
if (size <= 3) {
|
||||||
|
memcpy(&m.data, data, size);
|
||||||
|
return (m.value);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (utf8_put_big_item(data, size, &o) != 0)
|
||||||
|
goto fail;
|
||||||
|
m.data[0] = (o & 0xff);
|
||||||
|
m.data[1] = (o >> 8) & 0xff;
|
||||||
|
m.data[2] = (o >> 16);
|
||||||
|
return (m.value);
|
||||||
|
|
||||||
|
fail:
|
||||||
|
if (ud->width == 1)
|
||||||
|
return (utf8_big_space1.value);
|
||||||
|
return (utf8_big_space2.value);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get UTF-8 from index into buffer. */
|
||||||
|
void
|
||||||
|
utf8_get_big(uint32_t v, struct utf8_data *ud)
|
||||||
|
{
|
||||||
|
union utf8_big_map m = { .value = v };
|
||||||
|
struct utf8_big_item *bi;
|
||||||
|
u_int o;
|
||||||
|
|
||||||
|
memset(ud, 0, sizeof *ud);
|
||||||
|
ud->size = ud->have = (m.flags & UTF8_BIG_SIZE);
|
||||||
|
if (m.flags & UTF8_BIG_WIDTH2)
|
||||||
|
ud->width = 2;
|
||||||
|
else
|
||||||
|
ud->width = 1;
|
||||||
|
|
||||||
|
if (ud->size <= 3) {
|
||||||
|
memcpy(ud->data, m.data, ud->size);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
o = ((uint32_t)m.data[2] << 16)|((uint32_t)m.data[1] << 8)|m.data[0];
|
||||||
|
if (o >= utf8_big_list_used)
|
||||||
|
memset(ud->data, ' ', ud->size);
|
||||||
|
else {
|
||||||
|
bi = &utf8_big_list[o];
|
||||||
|
memcpy(ud->data, bi->data, ud->size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get big value for UTF-8 single character. */
|
||||||
|
uint32_t
|
||||||
|
utf8_set_big(char c, u_int width)
|
||||||
|
{
|
||||||
|
union utf8_big_map m = { .flags = 1, .data[0] = c };
|
||||||
|
|
||||||
|
if (width == 2)
|
||||||
|
m.flags |= UTF8_BIG_WIDTH2;
|
||||||
|
return (m.value);
|
||||||
|
}
|
||||||
|
|
||||||
/* Set a single character. */
|
/* Set a single character. */
|
||||||
void
|
void
|
||||||
utf8_set(struct utf8_data *ud, u_char ch)
|
utf8_set(struct utf8_data *ud, u_char ch)
|
||||||
|
@ -2551,23 +2551,33 @@ window_copy_search_rl_regex(struct grid *gd, u_int *ppx, u_int *psx, u_int py,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static const char *
|
static const char *
|
||||||
window_copy_cellstring(const struct grid_line *gl, u_int px, size_t *size)
|
window_copy_cellstring(const struct grid_line *gl, u_int px, size_t *size,
|
||||||
|
int *allocated)
|
||||||
{
|
{
|
||||||
|
static struct utf8_data ud;
|
||||||
struct grid_cell_entry *gce;
|
struct grid_cell_entry *gce;
|
||||||
|
char *copy;
|
||||||
|
|
||||||
if (px >= gl->cellsize) {
|
if (px >= gl->cellsize) {
|
||||||
*size = 1;
|
*size = 1;
|
||||||
|
*allocated = 0;
|
||||||
return (" ");
|
return (" ");
|
||||||
}
|
}
|
||||||
|
|
||||||
gce = &gl->celldata[px];
|
gce = &gl->celldata[px];
|
||||||
if (~gce->flags & GRID_FLAG_EXTENDED) {
|
if (~gce->flags & GRID_FLAG_EXTENDED) {
|
||||||
*size = 1;
|
*size = 1;
|
||||||
|
*allocated = 0;
|
||||||
return (&gce->data.data);
|
return (&gce->data.data);
|
||||||
}
|
}
|
||||||
|
|
||||||
*size = gl->extddata[gce->offset].data.size;
|
utf8_get_big(gl->extddata[gce->offset].data, &ud);
|
||||||
return (gl->extddata[gce->offset].data.data);
|
*size = ud.size;
|
||||||
|
*allocated = 1;
|
||||||
|
|
||||||
|
copy = xmalloc(ud.size);
|
||||||
|
memcpy(copy, ud.data, ud.size);
|
||||||
|
return (copy);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Find last match in given range. */
|
/* Find last match in given range. */
|
||||||
@ -2630,6 +2640,7 @@ window_copy_stringify(struct grid *gd, u_int py, u_int first, u_int last,
|
|||||||
const struct grid_line *gl;
|
const struct grid_line *gl;
|
||||||
const char *d;
|
const char *d;
|
||||||
size_t bufsize = 1024, dlen;
|
size_t bufsize = 1024, dlen;
|
||||||
|
int allocated;
|
||||||
|
|
||||||
while (bufsize < newsize)
|
while (bufsize < newsize)
|
||||||
bufsize *= 2;
|
bufsize *= 2;
|
||||||
@ -2638,7 +2649,7 @@ window_copy_stringify(struct grid *gd, u_int py, u_int first, u_int last,
|
|||||||
gl = grid_peek_line(gd, py);
|
gl = grid_peek_line(gd, py);
|
||||||
bx = *size - 1;
|
bx = *size - 1;
|
||||||
for (ax = first; ax < last; ax++) {
|
for (ax = first; ax < last; ax++) {
|
||||||
d = window_copy_cellstring(gl, ax, &dlen);
|
d = window_copy_cellstring(gl, ax, &dlen, &allocated);
|
||||||
newsize += dlen;
|
newsize += dlen;
|
||||||
while (bufsize < newsize) {
|
while (bufsize < newsize) {
|
||||||
bufsize *= 2;
|
bufsize *= 2;
|
||||||
@ -2650,6 +2661,8 @@ window_copy_stringify(struct grid *gd, u_int py, u_int first, u_int last,
|
|||||||
memcpy(buf + bx, d, dlen);
|
memcpy(buf + bx, d, dlen);
|
||||||
bx += dlen;
|
bx += dlen;
|
||||||
}
|
}
|
||||||
|
if (allocated)
|
||||||
|
free((void *)d);
|
||||||
}
|
}
|
||||||
buf[newsize - 1] = '\0';
|
buf[newsize - 1] = '\0';
|
||||||
|
|
||||||
@ -2670,6 +2683,7 @@ window_copy_cstrtocellpos(struct grid *gd, u_int ncells, u_int *ppx, u_int *ppy,
|
|||||||
struct {
|
struct {
|
||||||
const char *d;
|
const char *d;
|
||||||
size_t dlen;
|
size_t dlen;
|
||||||
|
int allocated;
|
||||||
} *cells;
|
} *cells;
|
||||||
|
|
||||||
/* Populate the array of cell data. */
|
/* Populate the array of cell data. */
|
||||||
@ -2680,7 +2694,7 @@ window_copy_cstrtocellpos(struct grid *gd, u_int ncells, u_int *ppx, u_int *ppy,
|
|||||||
gl = grid_peek_line(gd, pywrap);
|
gl = grid_peek_line(gd, pywrap);
|
||||||
while (cell < ncells) {
|
while (cell < ncells) {
|
||||||
cells[cell].d = window_copy_cellstring(gl, px,
|
cells[cell].d = window_copy_cellstring(gl, px,
|
||||||
&cells[cell].dlen);
|
&cells[cell].dlen, &cells[cell].allocated);
|
||||||
cell++;
|
cell++;
|
||||||
px++;
|
px++;
|
||||||
if (px == gd->sx) {
|
if (px == gd->sx) {
|
||||||
@ -2738,6 +2752,10 @@ window_copy_cstrtocellpos(struct grid *gd, u_int ncells, u_int *ppx, u_int *ppy,
|
|||||||
*ppy = pywrap;
|
*ppy = pywrap;
|
||||||
|
|
||||||
/* Free cell data. */
|
/* Free cell data. */
|
||||||
|
for (cell = 0; cell < ncells; cell++) {
|
||||||
|
if (cells[cell].allocated)
|
||||||
|
free((void *)cells[cell].d);
|
||||||
|
}
|
||||||
free(cells);
|
free(cells);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user