diff --git a/sbin/unwind/libunbound/util/netevent.c b/sbin/unwind/libunbound/util/netevent.c index 204e4883c..b9395a899 100644 --- a/sbin/unwind/libunbound/util/netevent.c +++ b/sbin/unwind/libunbound/util/netevent.c @@ -116,6 +116,8 @@ /** timeout in millisec to wait for write to unblock, packets dropped after.*/ #define SEND_BLOCKED_WAIT_TIMEOUT 200 +/** max number of times to wait for write to unblock, packets dropped after.*/ +#define SEND_BLOCKED_MAX_RETRY 5 /** Let's make timestamping code cleaner and redefine SO_TIMESTAMP* */ #ifndef SO_TIMESTAMP @@ -402,9 +404,10 @@ comm_point_send_udp_msg(struct comm_point *c, sldns_buffer* packet, WSAGetLastError() == WSAENOBUFS || WSAGetLastError() == WSAEWOULDBLOCK) { #endif + int retries = 0; /* if we set the fd blocking, other threads suddenly * have a blocking fd that they operate on */ - while(sent == -1 && ( + while(sent == -1 && retries < SEND_BLOCKED_MAX_RETRY && ( #ifndef USE_WINSOCK errno == EAGAIN || errno == EINTR || # ifdef EWOULDBLOCK @@ -419,6 +422,13 @@ comm_point_send_udp_msg(struct comm_point *c, sldns_buffer* packet, #endif )) { #if defined(HAVE_POLL) || defined(USE_WINSOCK) + int send_nobufs = ( +#ifndef USE_WINSOCK + errno == ENOBUFS +#else + WSAGetLastError() == WSAENOBUFS +#endif + ); struct pollfd p; int pret; memset(&p, 0, sizeof(p)); @@ -457,8 +467,48 @@ comm_point_send_udp_msg(struct comm_point *c, sldns_buffer* packet, log_err("poll udp out failed: %s", sock_strerror(errno)); return 0; + } else if((pret < 0 && +#ifndef USE_WINSOCK + errno == ENOBUFS +#else + WSAGetLastError() == WSAENOBUFS +#endif + ) || (send_nobufs && retries > 0)) { + /* ENOBUFS, and poll returned without + * a timeout. Or the retried send call + * returned ENOBUFS. It is good to + * wait a bit for the error to clear. */ + /* The timeout is 20*(2^(retries+1)), + * it increases exponentially, starting + * at 40 msec. After 5 tries, 1240 msec + * have passed in total, when poll + * returned the error, and 1200 msec + * when send returned the errors. */ +#ifndef USE_WINSOCK + pret = poll(NULL, 0, (SEND_BLOCKED_WAIT_TIMEOUT/10)<<(retries+1)); +#else + pret = WSAPoll(NULL, 0, (SEND_BLOCKED_WAIT_TIMEOUT/10)<<(retries+1)); +#endif + if(pret < 0 && +#ifndef USE_WINSOCK + errno != EAGAIN && errno != EINTR && +# ifdef EWOULDBLOCK + errno != EWOULDBLOCK && +# endif + errno != ENOBUFS +#else + WSAGetLastError() != WSAEINPROGRESS && + WSAGetLastError() != WSAEINTR && + WSAGetLastError() != WSAENOBUFS && + WSAGetLastError() != WSAEWOULDBLOCK +#endif + ) { + log_err("poll udp out timer failed: %s", + sock_strerror(errno)); + } } #endif /* defined(HAVE_POLL) || defined(USE_WINSOCK) */ + retries++; if (!is_connected) { sent = sendto(c->fd, (void*)sldns_buffer_begin(packet), sldns_buffer_remaining(packet), 0, @@ -665,7 +715,8 @@ comm_point_send_udp_msg_if(struct comm_point *c, sldns_buffer* packet, WSAGetLastError() == WSAENOBUFS || WSAGetLastError() == WSAEWOULDBLOCK) { #endif - while(sent == -1 && ( + int retries = 0; + while(sent == -1 && retries < SEND_BLOCKED_MAX_RETRY && ( #ifndef USE_WINSOCK errno == EAGAIN || errno == EINTR || # ifdef EWOULDBLOCK @@ -680,6 +731,13 @@ comm_point_send_udp_msg_if(struct comm_point *c, sldns_buffer* packet, #endif )) { #if defined(HAVE_POLL) || defined(USE_WINSOCK) + int send_nobufs = ( +#ifndef USE_WINSOCK + errno == ENOBUFS +#else + WSAGetLastError() == WSAENOBUFS +#endif + ); struct pollfd p; int pret; memset(&p, 0, sizeof(p)); @@ -718,8 +776,48 @@ comm_point_send_udp_msg_if(struct comm_point *c, sldns_buffer* packet, log_err("poll udp out failed: %s", sock_strerror(errno)); return 0; + } else if((pret < 0 && +#ifndef USE_WINSOCK + errno == ENOBUFS +#else + WSAGetLastError() == WSAENOBUFS +#endif + ) || (send_nobufs && retries > 0)) { + /* ENOBUFS, and poll returned without + * a timeout. Or the retried send call + * returned ENOBUFS. It is good to + * wait a bit for the error to clear. */ + /* The timeout is 20*(2^(retries+1)), + * it increases exponentially, starting + * at 40 msec. After 5 tries, 1240 msec + * have passed in total, when poll + * returned the error, and 1200 msec + * when send returned the errors. */ +#ifndef USE_WINSOCK + pret = poll(NULL, 0, (SEND_BLOCKED_WAIT_TIMEOUT/10)<<(retries+1)); +#else + pret = WSAPoll(NULL, 0, (SEND_BLOCKED_WAIT_TIMEOUT/10)<<(retries+1)); +#endif + if(pret < 0 && +#ifndef USE_WINSOCK + errno != EAGAIN && errno != EINTR && +# ifdef EWOULDBLOCK + errno != EWOULDBLOCK && +# endif + errno != ENOBUFS +#else + WSAGetLastError() != WSAEINPROGRESS && + WSAGetLastError() != WSAEINTR && + WSAGetLastError() != WSAENOBUFS && + WSAGetLastError() != WSAEWOULDBLOCK +#endif + ) { + log_err("poll udp out timer failed: %s", + sock_strerror(errno)); + } } #endif /* defined(HAVE_POLL) || defined(USE_WINSOCK) */ + retries++; sent = sendmsg(c->fd, &msg, 0); } } diff --git a/share/mk/bsd.regress.mk b/share/mk/bsd.regress.mk index 6f14dd1a3..1ab9e5947 100644 --- a/share/mk/bsd.regress.mk +++ b/share/mk/bsd.regress.mk @@ -1,4 +1,4 @@ -# $OpenBSD: bsd.regress.mk,v 1.25 2022/12/09 09:30:54 tb Exp $ +# $OpenBSD: bsd.regress.mk,v 1.26 2023/09/15 07:13:35 tb Exp $ # Documented in bsd.regress.mk(5) # No man pages for regression tests. @@ -21,7 +21,7 @@ all: regress REGRESS_LOG?=/dev/null REGRESS_SKIP_TARGETS?= REGRESS_SKIP_SLOW?=no -REGRESS_FAIL_EARLY?=no +REGRESS_FAIL_EARLY?=yes .if ! ${REGRESS_LOG:M/*} ERRORS += "Fatal: REGRESS_LOG=${REGRESS_LOG} is not an absolute path" diff --git a/sys/kern/kern_clockintr.c b/sys/kern/kern_clockintr.c index 11d44a936..2f1efcbb8 100644 --- a/sys/kern/kern_clockintr.c +++ b/sys/kern/kern_clockintr.c @@ -1,4 +1,4 @@ -/* $OpenBSD: kern_clockintr.c,v 1.52 2023/09/14 22:27:09 cheloha Exp $ */ +/* $OpenBSD: kern_clockintr.c,v 1.53 2023/09/15 11:48:49 deraadt Exp $ */ /* * Copyright (c) 2003 Dale Rahn * Copyright (c) 2020 Mark Kettenis @@ -355,9 +355,10 @@ clockintr_cancel(struct clockintr *cl) } struct clockintr * -clockintr_establish(struct cpu_info *ci, +clockintr_establish(void *vci, void (*func)(struct clockintr *, void *, void *), void *arg) { + struct cpu_info *ci = vci; struct clockintr *cl; struct clockintr_queue *cq = &ci->ci_queue; diff --git a/sys/sys/clockintr.h b/sys/sys/clockintr.h index 6b732f04b..20deb21d2 100644 --- a/sys/sys/clockintr.h +++ b/sys/sys/clockintr.h @@ -1,4 +1,4 @@ -/* $OpenBSD: clockintr.h,v 1.16 2023/09/14 22:07:11 cheloha Exp $ */ +/* $OpenBSD: clockintr.h,v 1.17 2023/09/15 11:48:48 deraadt Exp $ */ /* * Copyright (c) 2020-2022 Scott Cheloha * @@ -128,7 +128,7 @@ void clockintr_trigger(void); uint64_t clockintr_advance(struct clockintr *, uint64_t); uint64_t clockintr_advance_random(struct clockintr *, uint64_t, uint32_t); void clockintr_cancel(struct clockintr *); -struct clockintr *clockintr_establish(struct cpu_info *, +struct clockintr *clockintr_establish(void *, void (*)(struct clockintr *, void *, void *), void *); void clockintr_stagger(struct clockintr *, uint64_t, uint32_t, uint32_t); void clockqueue_init(struct clockintr_queue *); diff --git a/usr.bin/awk/awk.1 b/usr.bin/awk/awk.1 index e7c9dc2ba..7f8395d09 100644 --- a/usr.bin/awk/awk.1 +++ b/usr.bin/awk/awk.1 @@ -1,4 +1,4 @@ -.\" $OpenBSD: awk.1,v 1.63 2021/11/08 06:46:22 jmc Exp $ +.\" $OpenBSD: awk.1,v 1.64 2023/09/15 15:07:08 jsg Exp $ .\" .\" Copyright (C) Lucent Technologies 1997 .\" All Rights Reserved @@ -22,7 +22,7 @@ .\" ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF .\" THIS SOFTWARE. .\" -.Dd $Mdocdate: November 8 2021 $ +.Dd $Mdocdate: September 15 2023 $ .Dt AWK 1 .Os .Sh NAME @@ -1001,8 +1001,8 @@ No others do. .%A P. J. Weinberger .%T The AWK Programming Language .%I Addison-Wesley -.%D 1988 -.%O ISBN 0-201-07981-X +.%D 2024 +.%O ISBN 0-13-826972-6 .Re .Sh STANDARDS The diff --git a/usr.bin/tmux/cfg.c b/usr.bin/tmux/cfg.c index 286c00175..9da91532b 100644 --- a/usr.bin/tmux/cfg.c +++ b/usr.bin/tmux/cfg.c @@ -1,4 +1,4 @@ -/* $OpenBSD: cfg.c,v 1.86 2022/06/20 07:59:37 nicm Exp $ */ +/* $OpenBSD: cfg.c,v 1.87 2023/09/15 06:31:49 nicm Exp $ */ /* * Copyright (c) 2008 Nicholas Marriott @@ -67,6 +67,7 @@ start_cfg(void) { struct client *c; u_int i; + int flags = 0; /* * Configuration files are loaded without a client, so commands are run @@ -84,19 +85,17 @@ start_cfg(void) cmdq_append(c, cfg_item); } - for (i = 0; i < cfg_nfiles; i++) { - if (cfg_quiet) - load_cfg(cfg_files[i], c, NULL, CMD_PARSE_QUIET, NULL); - else - load_cfg(cfg_files[i], c, NULL, 0, NULL); - } + if (cfg_quiet) + flags = CMD_PARSE_QUIET; + for (i = 0; i < cfg_nfiles; i++) + load_cfg(cfg_files[i], c, NULL, NULL, flags, NULL); cmdq_append(NULL, cmdq_get_callback(cfg_done, NULL)); } int -load_cfg(const char *path, struct client *c, struct cmdq_item *item, int flags, - struct cmdq_item **new_item) +load_cfg(const char *path, struct client *c, struct cmdq_item *item, + struct cmd_find_state *current, int flags, struct cmdq_item **new_item) { FILE *f; struct cmd_parse_input pi; @@ -135,7 +134,7 @@ load_cfg(const char *path, struct client *c, struct cmdq_item *item, int flags, } if (item != NULL) - state = cmdq_copy_state(cmdq_get_state(item)); + state = cmdq_copy_state(cmdq_get_state(item), current); else state = cmdq_new_state(NULL, NULL, 0); cmdq_add_format(state, "current_file", "%s", pi.file); @@ -155,8 +154,8 @@ load_cfg(const char *path, struct client *c, struct cmdq_item *item, int flags, int load_cfg_from_buffer(const void *buf, size_t len, const char *path, - struct client *c, struct cmdq_item *item, int flags, - struct cmdq_item **new_item) + struct client *c, struct cmdq_item *item, struct cmd_find_state *current, + int flags, struct cmdq_item **new_item) { struct cmd_parse_input pi; struct cmd_parse_result *pr; @@ -187,7 +186,7 @@ load_cfg_from_buffer(const void *buf, size_t len, const char *path, } if (item != NULL) - state = cmdq_copy_state(cmdq_get_state(item)); + state = cmdq_copy_state(cmdq_get_state(item), current); else state = cmdq_new_state(NULL, NULL, 0); cmdq_add_format(state, "current_file", "%s", pi.file); diff --git a/usr.bin/tmux/cmd-queue.c b/usr.bin/tmux/cmd-queue.c index dc13d3a7b..3ce516803 100644 --- a/usr.bin/tmux/cmd-queue.c +++ b/usr.bin/tmux/cmd-queue.c @@ -1,4 +1,4 @@ -/* $OpenBSD: cmd-queue.c,v 1.114 2023/02/05 21:15:32 nicm Exp $ */ +/* $OpenBSD: cmd-queue.c,v 1.115 2023/09/15 06:31:49 nicm Exp $ */ /* * Copyright (c) 2013 Nicholas Marriott @@ -237,8 +237,10 @@ cmdq_link_state(struct cmdq_state *state) /* Make a copy of a state. */ struct cmdq_state * -cmdq_copy_state(struct cmdq_state *state) +cmdq_copy_state(struct cmdq_state *state, struct cmd_find_state *current) { + if (current != NULL) + return (cmdq_new_state(current, &state->event, state->flags)); return (cmdq_new_state(&state->current, &state->event, state->flags)); } diff --git a/usr.bin/tmux/cmd-source-file.c b/usr.bin/tmux/cmd-source-file.c index fde4225ae..83c29bf65 100644 --- a/usr.bin/tmux/cmd-source-file.c +++ b/usr.bin/tmux/cmd-source-file.c @@ -1,4 +1,4 @@ -/* $OpenBSD: cmd-source-file.c,v 1.53 2021/08/23 11:04:21 nicm Exp $ */ +/* $OpenBSD: cmd-source-file.c,v 1.54 2023/09/15 06:31:49 nicm Exp $ */ /* * Copyright (c) 2008 Tiago Cunha @@ -36,8 +36,10 @@ const struct cmd_entry cmd_source_file_entry = { .name = "source-file", .alias = "source", - .args = { "Fnqv", 1, -1, NULL }, - .usage = "[-Fnqv] path ...", + .args = { "t:Fnqv", 1, -1, NULL }, + .usage = "[-Fnqv] " CMD_TARGET_PANE_USAGE " path ...", + + .target = { 't', CMD_FIND_PANE, CMD_FIND_CANFAIL }, .flags = 0, .exec = cmd_source_file_exec @@ -93,6 +95,7 @@ cmd_source_file_done(struct client *c, const char *path, int error, size_t bsize = EVBUFFER_LENGTH(buffer); u_int n; struct cmdq_item *new_item; + struct cmd_find_state *target = cmdq_get_target(item); if (!closed) return; @@ -101,7 +104,7 @@ cmd_source_file_done(struct client *c, const char *path, int error, cmdq_error(item, "%s: %s", path, strerror(error)); else if (bsize != 0) { if (load_cfg_from_buffer(bdata, bsize, path, c, cdata->after, - cdata->flags, &new_item) < 0) + target, cdata->flags, &new_item) < 0) cdata->retval = CMD_RETURN_ERROR; else if (new_item != NULL) cdata->after = new_item; diff --git a/usr.bin/tmux/input.c b/usr.bin/tmux/input.c index 41826662b..b6396a9cb 100644 --- a/usr.bin/tmux/input.c +++ b/usr.bin/tmux/input.c @@ -1,4 +1,4 @@ -/* $OpenBSD: input.c,v 1.220 2023/08/23 08:30:07 nicm Exp $ */ +/* $OpenBSD: input.c,v 1.222 2023/09/15 06:31:49 nicm Exp $ */ /* * Copyright (c) 2007 Nicholas Marriott @@ -169,6 +169,7 @@ static void input_csi_dispatch_rm(struct input_ctx *); static void input_csi_dispatch_rm_private(struct input_ctx *); static void input_csi_dispatch_sm(struct input_ctx *); static void input_csi_dispatch_sm_private(struct input_ctx *); +static void input_csi_dispatch_sm_graphics(struct input_ctx *); static void input_csi_dispatch_winops(struct input_ctx *); static void input_csi_dispatch_sgr_256(struct input_ctx *, int, u_int *); static void input_csi_dispatch_sgr_rgb(struct input_ctx *, int, u_int *); @@ -203,7 +204,7 @@ enum input_esc_type { INPUT_ESC_SCSG0_ON, INPUT_ESC_SCSG1_OFF, INPUT_ESC_SCSG1_ON, - INPUT_ESC_ST, + INPUT_ESC_ST }; /* Escape command table. */ @@ -259,11 +260,12 @@ enum input_csi_type { INPUT_CSI_SGR, INPUT_CSI_SM, INPUT_CSI_SM_PRIVATE, + INPUT_CSI_SM_GRAPHICS, INPUT_CSI_SU, INPUT_CSI_TBC, INPUT_CSI_VPA, INPUT_CSI_WINOPS, - INPUT_CSI_XDA, + INPUT_CSI_XDA }; /* Control (CSI) command table. */ @@ -283,6 +285,7 @@ static const struct input_table_entry input_csi_table[] = { { 'M', "", INPUT_CSI_DL }, { 'P', "", INPUT_CSI_DCH }, { 'S', "", INPUT_CSI_SU }, + { 'S', "?", INPUT_CSI_SM_GRAPHICS }, { 'T', "", INPUT_CSI_SD }, { 'X', "", INPUT_CSI_ECH }, { 'Z', "", INPUT_CSI_CBT }, @@ -306,7 +309,7 @@ static const struct input_table_entry input_csi_table[] = { { 'r', "", INPUT_CSI_DECSTBM }, { 's', "", INPUT_CSI_SCP }, { 't', "", INPUT_CSI_WINOPS }, - { 'u', "", INPUT_CSI_RCP }, + { 'u', "", INPUT_CSI_RCP } }; /* Input transition. */ @@ -1595,6 +1598,9 @@ input_csi_dispatch(struct input_ctx *ictx) case INPUT_CSI_SM_PRIVATE: input_csi_dispatch_sm_private(ictx); break; + case INPUT_CSI_SM_GRAPHICS: + input_csi_dispatch_sm_graphics(ictx); + break; case INPUT_CSI_SU: n = input_get(ictx, 0, 1, 1); if (n != -1) @@ -1827,6 +1833,12 @@ input_csi_dispatch_sm_private(struct input_ctx *ictx) } } +/* Handle CSI graphics SM. */ +static void +input_csi_dispatch_sm_graphics(__unused struct input_ctx *ictx) +{ +} + /* Handle CSI window operations. */ static void input_csi_dispatch_winops(struct input_ctx *ictx) @@ -1834,6 +1846,7 @@ input_csi_dispatch_winops(struct input_ctx *ictx) struct screen_write_ctx *sctx = &ictx->ctx; struct screen *s = sctx->s; struct window_pane *wp = ictx->wp; + struct window *w = wp->window; u_int x = screen_size_x(s), y = screen_size_y(s); int n, m; @@ -1847,8 +1860,6 @@ input_csi_dispatch_winops(struct input_ctx *ictx) case 7: case 11: case 13: - case 14: - case 19: case 20: case 21: case 24: @@ -1866,6 +1877,21 @@ input_csi_dispatch_winops(struct input_ctx *ictx) if (input_get(ictx, m, 0, -1) == -1) return; break; + case 14: + input_reply(ictx, "\033[4;%u;%ut", y * w->ypixel, x * w->xpixel); + break; + case 15: + input_reply(ictx, "\033[5;%u;%ut", y * w->ypixel, x * w->xpixel); + break; + case 16: + input_reply(ictx, "\033[6;%u;%ut", w->ypixel, w->xpixel); + break; + case 18: + input_reply(ictx, "\033[8;%u;%ut", y, x); + break; + case 19: + input_reply(ictx, "\033[9;%u;%ut", y, x); + break; case 22: m++; switch (input_get(ictx, m, 0, -1)) { @@ -1893,9 +1919,6 @@ input_csi_dispatch_winops(struct input_ctx *ictx) break; } break; - case 18: - input_reply(ictx, "\033[8;%u;%ut", y, x); - break; default: log_debug("%s: unknown '%c'", __func__, ictx->ch); break; diff --git a/usr.bin/tmux/screen-write.c b/usr.bin/tmux/screen-write.c index 96a4916be..08d643843 100644 --- a/usr.bin/tmux/screen-write.c +++ b/usr.bin/tmux/screen-write.c @@ -1,4 +1,4 @@ -/* $OpenBSD: screen-write.c,v 1.221 2023/09/14 13:01:35 nicm Exp $ */ +/* $OpenBSD: screen-write.c,v 1.222 2023/09/15 15:49:05 nicm Exp $ */ /* * Copyright (c) 2007 Nicholas Marriott @@ -32,8 +32,8 @@ static void screen_write_collect_flush(struct screen_write_ctx *, int, const char *); static int screen_write_overwrite(struct screen_write_ctx *, struct grid_cell *, u_int); -static const struct grid_cell *screen_write_combine(struct screen_write_ctx *, - const struct utf8_data *, u_int *, u_int *); +static int screen_write_combine(struct screen_write_ctx *, + const struct grid_cell *); struct screen_write_citem { u_int x; @@ -1742,7 +1742,6 @@ screen_write_collect_end(struct screen_write_ctx *ctx) if (ci->used == 0) return; - ctx->flags &= ~SCREEN_WRITE_COMBINE; before = screen_write_collect_trim(ctx, s->cy, s->cx, ci->used, &wrapped); @@ -1841,65 +1840,22 @@ screen_write_cell(struct screen_write_ctx *ctx, const struct grid_cell *gc) { struct screen *s = ctx->s; struct grid *gd = s->grid; - struct grid_cell copy; - const struct utf8_data *ud = &gc->data, *previous = NULL, *combine; + const struct utf8_data *ud = &gc->data; struct grid_line *gl; struct grid_cell_entry *gce; struct grid_cell tmp_gc, now_gc; struct tty_ctx ttyctx; u_int sx = screen_size_x(s), sy = screen_size_y(s); - u_int width = ud->width, xx, last, cx, cy; + u_int width = ud->width, xx, not_wrap; int selected, skip = 1; /* Ignore padding cells. */ if (gc->flags & GRID_FLAG_PADDING) return; - /* Check if this cell needs to be combined with the previous cell. */ - if (ctx->flags & SCREEN_WRITE_COMBINE) - previous = &ctx->previous; - switch (utf8_try_combined(ud, previous, &combine, &width)) { - case UTF8_DISCARD_NOW: - log_debug("%s: UTF8_DISCARD_NOW (width %u)", __func__, width); - ctx->flags &= ~SCREEN_WRITE_COMBINE; + /* Get the previous cell to check for combining. */ + if (screen_write_combine(ctx, gc) != 0) return; - case UTF8_WRITE_NOW: - log_debug("%s: UTF8_WRITE_NOW (width %u)", __func__, width); - ctx->flags &= ~SCREEN_WRITE_COMBINE; - break; - case UTF8_COMBINE_NOW: - log_debug("%s: UTF8_COMBINE_NOW (width %u)", __func__, width); - screen_write_collect_flush(ctx, 0, __func__); - gc = screen_write_combine(ctx, combine, &xx, &cx); - if (gc != NULL) { - cx = s->cx; cy = s->cy; - screen_write_set_cursor(ctx, xx, s->cy); - screen_write_initctx(ctx, &ttyctx, 0); - ttyctx.cell = gc; - tty_write(tty_cmd_cell, &ttyctx); - s->cx = cx; s->cy = cy; - } - ctx->flags &= ~SCREEN_WRITE_COMBINE; - return; - case UTF8_WRITE_MAYBE_COMBINE: - log_debug("%s: UTF8_WRITE_MAYBE_COMBINE (width %u)", __func__, - width); - utf8_copy(&ctx->previous, ud); - ctx->flags |= SCREEN_WRITE_COMBINE; - break; - case UTF8_DISCARD_MAYBE_COMBINE: - log_debug("%s: UTF8_DISCARD_MAYBE_COMBINE (width %u)", __func__, - width); - utf8_copy(&ctx->previous, ud); - ctx->flags |= SCREEN_WRITE_COMBINE; - return; - } - if (width != ud->width) { - memcpy(©, gc, sizeof copy); - copy.data.width = width; - gc = © - } - ud = NULL; /* Flush any existing scrolling. */ screen_write_collect_flush(ctx, 1, __func__); @@ -1991,11 +1947,11 @@ screen_write_cell(struct screen_write_ctx *ctx, const struct grid_cell *gc) * Move the cursor. If not wrapping, stick at the last character and * replace it. */ - last = !(s->mode & MODE_WRAP); - if (s->cx <= sx - last - width) + not_wrap = !(s->mode & MODE_WRAP); + if (s->cx <= sx - not_wrap - width) screen_write_set_cursor(ctx, s->cx + width, -1); else - screen_write_set_cursor(ctx, sx - last, -1); + screen_write_set_cursor(ctx, sx - not_wrap, -1); /* Create space for character in insert mode. */ if (s->mode & MODE_INSERT) { @@ -2015,65 +1971,98 @@ screen_write_cell(struct screen_write_ctx *ctx, const struct grid_cell *gc) } } -/* Combine a UTF-8 zero-width character onto the previous. */ -static const struct grid_cell * -screen_write_combine(struct screen_write_ctx *ctx, const struct utf8_data *ud, - u_int *xx, u_int *cx) +/* Combine a UTF-8 zero-width character onto the previous if necessary. */ +static int +screen_write_combine(struct screen_write_ctx *ctx, const struct grid_cell *gc) { struct screen *s = ctx->s; struct grid *gd = s->grid; - static struct grid_cell gc; - u_int n, width; + const struct utf8_data *ud = &gc->data; + u_int n, cx = s->cx, cy = s->cy; + struct grid_cell last; + struct tty_ctx ttyctx; + int force_wide = 0, zero_width = 0; - /* Can't combine if at 0. */ - if (s->cx == 0) { - *xx = 0; - return (NULL); + /* + * Is this character which makes no sense without being combined? If + * this is true then flag it here and discard the character (return 1) + * if we cannot combine it. + */ + if (utf8_is_zwj(ud)) + zero_width = 1; + else if (utf8_is_vs(ud)) + zero_width = force_wide = 1; + else if (ud->width == 0) + zero_width = 1; + + /* Cannot combine empty character or at left. */ + if (ud->size < 2 || cx == 0) + return (zero_width); + log_debug("%s: character %.*s at %u,%u (width %u)", __func__, + (int)ud->size, ud->data, cx, cy, ud->width); + + /* Find the cell to combine with. */ + n = 1; + grid_view_get_cell(gd, cx - n, cy, &last); + if (cx != 1 && (last.flags & GRID_FLAG_PADDING)) { + n = 2; + grid_view_get_cell(gd, cx - n, cy, &last); } - *xx = s->cx; + if (n != last.data.width || (last.flags & GRID_FLAG_PADDING)) + return (zero_width); - /* Empty data is out. */ - if (ud->size == 0) - fatalx("UTF-8 data empty"); - - /* Retrieve the previous cell. */ - for (n = 1; n <= s->cx; n++) { - grid_view_get_cell(gd, s->cx - n, s->cy, &gc); - if (~gc.flags & GRID_FLAG_PADDING) - break; + /* + * Check if we need to combine characters. This could be zero width + * (zet above), a modifier character (with an existing Unicode + * character) or a previous ZWJ. + */ + if (!zero_width) { + if (utf8_is_modifier(ud)) { + if (last.data.size < 2) + return (0); + force_wide = 1; + } else if (!utf8_has_zwj(&last.data)) + return (0); } - if (n > s->cx) - return (NULL); - /* Check there is enough space. */ - if (gc.data.size + ud->size > sizeof gc.data.data) - return (NULL); - (*xx) -= n; + /* Combining; flush any pending output. */ + screen_write_collect_flush(ctx, 0, __func__); - log_debug("%s: %.*s onto %.*s at %u,%u (width %u)", __func__, - (int)ud->size, ud->data, (int)gc.data.size, gc.data.data, *xx, - s->cy, gc.data.width); + log_debug("%s: %.*s -> %.*s at %u,%u (offset %u, width %u)", __func__, + (int)ud->size, ud->data, (int)last.data.size, last.data.data, + cx - n, cy, n, last.data.width); /* Append the data. */ - memcpy(gc.data.data + gc.data.size, ud->data, ud->size); - gc.data.size += ud->size; - width = gc.data.width; + memcpy(last.data.data + last.data.size, ud->data, ud->size); + last.data.size += ud->size; - /* If this is U+FE0F VARIATION SELECTOR-16, force the width to 2. */ - if (gc.data.width == 1 && - ud->size == 3 && - memcmp(ud->data, "\357\270\217", 3) == 0) { - grid_view_set_padding(gd, (*xx) + 1, s->cy); - gc.data.width = 2; - width += 2; - } + /* Force the width to 2 for modifiers and variation selector. */ + if (last.data.width == 1 && force_wide) { + last.data.width = 2; + n = 2; + cx++; + } else + force_wide = 0; /* Set the new cell. */ - grid_view_set_cell(gd, *xx, s->cy, &gc); + grid_view_set_cell(gd, cx - n, cy, &last); + if (force_wide) + grid_view_set_padding(gd, cx, cy); - *cx = (*xx) + width; - log_debug("%s: character at %u; cursor at %u", __func__, *xx, *cx); - return (&gc); + /* + * Redraw the combined cell. If forcing the cell to width 2, reset the + * cached cursor position in the tty, since we don't really know + * whether the terminal thought the character was width 1 or width 2 + * and what it is going to do now. + */ + screen_write_set_cursor(ctx, cx - n, cy); + screen_write_initctx(ctx, &ttyctx, 0); + ttyctx.cell = &last; + ttyctx.num = force_wide; /* reset cached cursor position */ + tty_write(tty_cmd_cell, &ttyctx); + screen_write_set_cursor(ctx, cx, cy); + + return (1); } /* diff --git a/usr.bin/tmux/server.c b/usr.bin/tmux/server.c index c9d116f93..494a6231e 100644 --- a/usr.bin/tmux/server.c +++ b/usr.bin/tmux/server.c @@ -1,4 +1,4 @@ -/* $OpenBSD: server.c,v 1.204 2023/09/01 14:29:11 nicm Exp $ */ +/* $OpenBSD: server.c,v 1.205 2023/09/15 15:49:05 nicm Exp $ */ /* * Copyright (c) 2007 Nicholas Marriott @@ -205,7 +205,6 @@ server_start(struct tmuxproc *client, int flags, struct event_base *base, fatal("pledge failed"); input_key_build(); - utf8_build_combined(); RB_INIT(&windows); RB_INIT(&all_window_panes); TAILQ_INIT(&clients); diff --git a/usr.bin/tmux/tmux.1 b/usr.bin/tmux/tmux.1 index 4a33de855..5f6bc7d9f 100644 --- a/usr.bin/tmux/tmux.1 +++ b/usr.bin/tmux/tmux.1 @@ -1,4 +1,4 @@ -.\" $OpenBSD: tmux.1,v 1.931 2023/09/02 09:17:23 nicm Exp $ +.\" $OpenBSD: tmux.1,v 1.932 2023/09/15 06:31:49 nicm Exp $ .\" .\" Copyright (c) 2007 Nicholas Marriott .\" @@ -14,7 +14,7 @@ .\" IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING .\" OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. .\" -.Dd $Mdocdate: September 2 2023 $ +.Dd $Mdocdate: September 15 2023 $ .Dt TMUX 1 .Os .Sh NAME @@ -1546,6 +1546,7 @@ show debugging information about jobs and terminals. .Tg source .It Xo Ic source-file .Op Fl Fnqv +.Op Fl t Ar target-pane .Ar path ... .Xc .D1 Pq alias: Ic source diff --git a/usr.bin/tmux/tmux.h b/usr.bin/tmux/tmux.h index 74a86482a..43bf574e5 100644 --- a/usr.bin/tmux/tmux.h +++ b/usr.bin/tmux/tmux.h @@ -1,4 +1,4 @@ -/* $OpenBSD: tmux.h,v 1.1209 2023/09/02 20:03:10 nicm Exp $ */ +/* $OpenBSD: tmux.h,v 1.1211 2023/09/15 15:49:05 nicm Exp $ */ /* * Copyright (c) 2007 Nicholas Marriott @@ -30,6 +30,7 @@ #include #include #include +#include #include "tmux-protocol.h" #include "xmalloc.h" @@ -619,15 +620,6 @@ enum utf8_state { UTF8_ERROR }; -/* UTF-8 combine state. */ -enum utf8_combine_state { - UTF8_DISCARD_NOW, /* discard immediately */ - UTF8_WRITE_NOW, /* do not combine, write immediately */ - UTF8_COMBINE_NOW, /* combine immediately */ - UTF8_WRITE_MAYBE_COMBINE, /* write but try to combine the next */ - UTF8_DISCARD_MAYBE_COMBINE /* discard but try to combine the next */ -}; - /* Colour flags. */ #define COLOUR_FLAG_256 0x01000000 #define COLOUR_FLAG_RGB 0x02000000 @@ -900,7 +892,6 @@ struct screen_write_ctx { int flags; #define SCREEN_WRITE_SYNC 0x1 -#define SCREEN_WRITE_COMBINE 0x2 screen_write_init_ctx_cb init_ctx_cb; void *arg; @@ -908,7 +899,6 @@ struct screen_write_ctx { struct screen_write_citem *item; u_int scrolled; u_int bg; - struct utf8_data previous; }; /* Box border lines option. */ @@ -2102,10 +2092,11 @@ extern char **cfg_files; extern u_int cfg_nfiles; extern int cfg_quiet; void start_cfg(void); -int load_cfg(const char *, struct client *, struct cmdq_item *, int, - struct cmdq_item **); +int load_cfg(const char *, struct client *, struct cmdq_item *, + struct cmd_find_state *, int, struct cmdq_item **); int load_cfg_from_buffer(const void *, size_t, const char *, - struct client *, struct cmdq_item *, int, struct cmdq_item **); + struct client *, struct cmdq_item *, struct cmd_find_state *, + int, struct cmdq_item **); void printflike(1, 2) cfg_add_cause(const char *, ...); void cfg_print_causes(struct cmdq_item *); void cfg_show_causes(struct session *); @@ -2551,7 +2542,8 @@ struct cmd_parse_result *cmd_parse_from_arguments(struct args_value *, u_int, struct cmdq_state *cmdq_new_state(struct cmd_find_state *, struct key_event *, int); struct cmdq_state *cmdq_link_state(struct cmdq_state *); -struct cmdq_state *cmdq_copy_state(struct cmdq_state *); +struct cmdq_state *cmdq_copy_state(struct cmdq_state *, + struct cmd_find_state *); void cmdq_free_state(struct cmdq_state *); void printflike(3, 4) cmdq_add_format(struct cmdq_state *, const char *, const char *, ...); @@ -3275,6 +3267,8 @@ u_int session_group_attached_count(struct session_group *); void session_renumber_windows(struct session *); /* utf8.c */ +enum utf8_state utf8_towc (const struct utf8_data *, wchar_t *); +int utf8_in_table(wchar_t, const wchar_t *, u_int); utf8_char utf8_build_one(u_char); enum utf8_state utf8_from_data(const struct utf8_data *, utf8_char *); void utf8_to_data(utf8_char, struct utf8_data *); @@ -3297,10 +3291,10 @@ char *utf8_rpadcstr(const char *, u_int); int utf8_cstrhas(const char *, const struct utf8_data *); /* utf8-combined.c */ -void utf8_build_combined(void); -int utf8_try_combined(const struct utf8_data *, - const struct utf8_data *, const struct utf8_data **, - u_int *width); +int utf8_has_zwj(const struct utf8_data *); +int utf8_is_zwj(const struct utf8_data *); +int utf8_is_vs(const struct utf8_data *); +int utf8_is_modifier(const struct utf8_data *); /* procname.c */ char *get_proc_name(int, char *); diff --git a/usr.bin/tmux/tty.c b/usr.bin/tmux/tty.c index efbc4a3cc..e6205c8fc 100644 --- a/usr.bin/tmux/tty.c +++ b/usr.bin/tmux/tty.c @@ -1,4 +1,4 @@ -/* $OpenBSD: tty.c,v 1.434 2023/09/02 20:03:10 nicm Exp $ */ +/* $OpenBSD: tty.c,v 1.435 2023/09/15 15:49:05 nicm Exp $ */ /* * Copyright (c) 2007 Nicholas Marriott @@ -2091,6 +2091,9 @@ tty_cmd_cell(struct tty *tty, const struct tty_ctx *ctx) tty_cell(tty, ctx->cell, &ctx->defaults, ctx->palette, ctx->s->hyperlinks); + + if (ctx->num == 1) + tty_invalidate(tty); } void diff --git a/usr.bin/tmux/utf8-combined.c b/usr.bin/tmux/utf8-combined.c index 18dffbf6f..1f0e2443b 100644 --- a/usr.bin/tmux/utf8-combined.c +++ b/usr.bin/tmux/utf8-combined.c @@ -1,4 +1,4 @@ -/* $OpenBSD: utf8-combined.c,v 1.2 2023/09/01 18:43:54 nicm Exp $ */ +/* $OpenBSD: utf8-combined.c,v 1.3 2023/09/15 15:49:05 nicm Exp $ */ /* * Copyright (c) 2023 Nicholas Marriott @@ -24,1112 +24,77 @@ #include "tmux.h" -static const struct { - wchar_t first; - wchar_t second; -} utf8_combined_table[] = { - { 0x1F1E6, 0x1F1E8 }, /* flag: Ascension Island */ - { 0x1F1E6, 0x1F1E9 }, /* flag: Andorra */ - { 0x1F1E6, 0x1F1EA }, /* flag: United Arab Emirates */ - { 0x1F1E6, 0x1F1EB }, /* flag: Afghanistan */ - { 0x1F1E6, 0x1F1EC }, /* flag: Antigua & Barbuda */ - { 0x1F1E6, 0x1F1EE }, /* flag: Anguilla */ - { 0x1F1E6, 0x1F1F1 }, /* flag: Albania */ - { 0x1F1E6, 0x1F1F2 }, /* flag: Armenia */ - { 0x1F1E6, 0x1F1F4 }, /* flag: Angola */ - { 0x1F1E6, 0x1F1F6 }, /* flag: Antarctica */ - { 0x1F1E6, 0x1F1F7 }, /* flag: Argentina */ - { 0x1F1E6, 0x1F1F8 }, /* flag: American Samoa */ - { 0x1F1E6, 0x1F1F9 }, /* flag: Austria */ - { 0x1F1E6, 0x1F1FA }, /* flag: Australia */ - { 0x1F1E6, 0x1F1FC }, /* flag: Aruba */ - { 0x1F1E6, 0x1F1FD }, /* flag: Aland Islands */ - { 0x1F1E6, 0x1F1FF }, /* flag: Azerbaijan */ - { 0x1F1E7, 0x1F1E6 }, /* flag: Bosnia & Herzegovina */ - { 0x1F1E7, 0x1F1E7 }, /* flag: Barbados */ - { 0x1F1E7, 0x1F1E9 }, /* flag: Bangladesh */ - { 0x1F1E7, 0x1F1EA }, /* flag: Belgium */ - { 0x1F1E7, 0x1F1EB }, /* flag: Burkina Faso */ - { 0x1F1E7, 0x1F1EC }, /* flag: Bulgaria */ - { 0x1F1E7, 0x1F1ED }, /* flag: Bahrain */ - { 0x1F1E7, 0x1F1EE }, /* flag: Burundi */ - { 0x1F1E7, 0x1F1EF }, /* flag: Benin */ - { 0x1F1E7, 0x1F1F1 }, /* flag: St. Barthelemy */ - { 0x1F1E7, 0x1F1F2 }, /* flag: Bermuda */ - { 0x1F1E7, 0x1F1F3 }, /* flag: Brunei */ - { 0x1F1E7, 0x1F1F4 }, /* flag: Bolivia */ - { 0x1F1E7, 0x1F1F6 }, /* flag: Caribbean Netherlands */ - { 0x1F1E7, 0x1F1F7 }, /* flag: Brazil */ - { 0x1F1E7, 0x1F1F8 }, /* flag: Bahamas */ - { 0x1F1E7, 0x1F1F9 }, /* flag: Bhutan */ - { 0x1F1E7, 0x1F1FB }, /* flag: Bouvet Island */ - { 0x1F1E7, 0x1F1FC }, /* flag: Botswana */ - { 0x1F1E7, 0x1F1FE }, /* flag: Belarus */ - { 0x1F1E7, 0x1F1FF }, /* flag: Belize */ - { 0x1F1E8, 0x1F1E6 }, /* flag: Canada */ - { 0x1F1E8, 0x1F1E8 }, /* flag: Cocos (Keeling) Islands */ - { 0x1F1E8, 0x1F1E9 }, /* flag: Congo - Kinshasa */ - { 0x1F1E8, 0x1F1EB }, /* flag: Central African Republic */ - { 0x1F1E8, 0x1F1EC }, /* flag: Congo - Brazzaville */ - { 0x1F1E8, 0x1F1ED }, /* flag: Switzerland */ - { 0x1F1E8, 0x1F1EE }, /* flag: Cote d'Ivoire */ - { 0x1F1E8, 0x1F1F0 }, /* flag: Cook Islands */ - { 0x1F1E8, 0x1F1F1 }, /* flag: Chile */ - { 0x1F1E8, 0x1F1F2 }, /* flag: Cameroon */ - { 0x1F1E8, 0x1F1F3 }, /* flag: China */ - { 0x1F1E8, 0x1F1F4 }, /* flag: Colombia */ - { 0x1F1E8, 0x1F1F5 }, /* flag: Clipperton Island */ - { 0x1F1E8, 0x1F1F7 }, /* flag: Costa Rica */ - { 0x1F1E8, 0x1F1FA }, /* flag: Cuba */ - { 0x1F1E8, 0x1F1FB }, /* flag: Cape Verde */ - { 0x1F1E8, 0x1F1FC }, /* flag: Curacao */ - { 0x1F1E8, 0x1F1FD }, /* flag: Christmas Island */ - { 0x1F1E8, 0x1F1FE }, /* flag: Cyprus */ - { 0x1F1E8, 0x1F1FF }, /* flag: Czechia */ - { 0x1F1E9, 0x1F1EA }, /* flag: Germany */ - { 0x1F1E9, 0x1F1EC }, /* flag: Diego Garcia */ - { 0x1F1E9, 0x1F1EF }, /* flag: Djibouti */ - { 0x1F1E9, 0x1F1F0 }, /* flag: Denmark */ - { 0x1F1E9, 0x1F1F2 }, /* flag: Dominica */ - { 0x1F1E9, 0x1F1F4 }, /* flag: Dominican Republic */ - { 0x1F1E9, 0x1F1FF }, /* flag: Algeria */ - { 0x1F1EA, 0x1F1E6 }, /* flag: Ceuta & Melilla */ - { 0x1F1EA, 0x1F1E8 }, /* flag: Ecuador */ - { 0x1F1EA, 0x1F1EA }, /* flag: Estonia */ - { 0x1F1EA, 0x1F1EC }, /* flag: Egypt */ - { 0x1F1EA, 0x1F1ED }, /* flag: Western Sahara */ - { 0x1F1EA, 0x1F1F7 }, /* flag: Eritrea */ - { 0x1F1EA, 0x1F1F8 }, /* flag: Spain */ - { 0x1F1EA, 0x1F1F9 }, /* flag: Ethiopia */ - { 0x1F1EA, 0x1F1FA }, /* flag: European Union */ - { 0x1F1EB, 0x1F1EE }, /* flag: Finland */ - { 0x1F1EB, 0x1F1EF }, /* flag: Fiji */ - { 0x1F1EB, 0x1F1F0 }, /* flag: Falkland Islands */ - { 0x1F1EB, 0x1F1F2 }, /* flag: Micronesia */ - { 0x1F1EB, 0x1F1F4 }, /* flag: Faroe Islands */ - { 0x1F1EB, 0x1F1F7 }, /* flag: France */ - { 0x1F1EC, 0x1F1E6 }, /* flag: Gabon */ - { 0x1F1EC, 0x1F1E7 }, /* flag: United Kingdom */ - { 0x1F1EC, 0x1F1E9 }, /* flag: Grenada */ - { 0x1F1EC, 0x1F1EA }, /* flag: Georgia */ - { 0x1F1EC, 0x1F1EB }, /* flag: French Guiana */ - { 0x1F1EC, 0x1F1EC }, /* flag: Guernsey */ - { 0x1F1EC, 0x1F1ED }, /* flag: Ghana */ - { 0x1F1EC, 0x1F1EE }, /* flag: Gibraltar */ - { 0x1F1EC, 0x1F1F1 }, /* flag: Greenland */ - { 0x1F1EC, 0x1F1F2 }, /* flag: Gambia */ - { 0x1F1EC, 0x1F1F3 }, /* flag: Guinea */ - { 0x1F1EC, 0x1F1F5 }, /* flag: Guadeloupe */ - { 0x1F1EC, 0x1F1F6 }, /* flag: Equatorial Guinea */ - { 0x1F1EC, 0x1F1F7 }, /* flag: Greece */ - { 0x1F1EC, 0x1F1F8 }, /* flag: South Georgia & South Sandwich Islands */ - { 0x1F1EC, 0x1F1F9 }, /* flag: Guatemala */ - { 0x1F1EC, 0x1F1FA }, /* flag: Guam */ - { 0x1F1EC, 0x1F1FC }, /* flag: Guinea-Bissau */ - { 0x1F1EC, 0x1F1FE }, /* flag: Guyana */ - { 0x1F1ED, 0x1F1F0 }, /* flag: Hong Kong SAR China */ - { 0x1F1ED, 0x1F1F2 }, /* flag: Heard & McDonald Islands */ - { 0x1F1ED, 0x1F1F3 }, /* flag: Honduras */ - { 0x1F1ED, 0x1F1F7 }, /* flag: Croatia */ - { 0x1F1ED, 0x1F1F9 }, /* flag: Haiti */ - { 0x1F1ED, 0x1F1FA }, /* flag: Hungary */ - { 0x1F1EE, 0x1F1E8 }, /* flag: Canary Islands */ - { 0x1F1EE, 0x1F1E9 }, /* flag: Indonesia */ - { 0x1F1EE, 0x1F1EA }, /* flag: Ireland */ - { 0x1F1EE, 0x1F1F1 }, /* flag: Israel */ - { 0x1F1EE, 0x1F1F2 }, /* flag: Isle of Man */ - { 0x1F1EE, 0x1F1F3 }, /* flag: India */ - { 0x1F1EE, 0x1F1F4 }, /* flag: British Indian Ocean Territory */ - { 0x1F1EE, 0x1F1F6 }, /* flag: Iraq */ - { 0x1F1EE, 0x1F1F7 }, /* flag: Iran */ - { 0x1F1EE, 0x1F1F8 }, /* flag: Iceland */ - { 0x1F1EE, 0x1F1F9 }, /* flag: Italy */ - { 0x1F1EF, 0x1F1EA }, /* flag: Jersey */ - { 0x1F1EF, 0x1F1F2 }, /* flag: Jamaica */ - { 0x1F1EF, 0x1F1F4 }, /* flag: Jordan */ - { 0x1F1EF, 0x1F1F5 }, /* flag: Japan */ - { 0x1F1F0, 0x1F1EA }, /* flag: Kenya */ - { 0x1F1F0, 0x1F1EC }, /* flag: Kyrgyzstan */ - { 0x1F1F0, 0x1F1ED }, /* flag: Cambodia */ - { 0x1F1F0, 0x1F1EE }, /* flag: Kiribati */ - { 0x1F1F0, 0x1F1F2 }, /* flag: Comoros */ - { 0x1F1F0, 0x1F1F3 }, /* flag: St. Kitts & Nevis */ - { 0x1F1F0, 0x1F1F5 }, /* flag: North Korea */ - { 0x1F1F0, 0x1F1F7 }, /* flag: South Korea */ - { 0x1F1F0, 0x1F1FC }, /* flag: Kuwait */ - { 0x1F1F0, 0x1F1FE }, /* flag: Cayman Islands */ - { 0x1F1F0, 0x1F1FF }, /* flag: Kazakhstan */ - { 0x1F1F1, 0x1F1E6 }, /* flag: Laos */ - { 0x1F1F1, 0x1F1E7 }, /* flag: Lebanon */ - { 0x1F1F1, 0x1F1E8 }, /* flag: St. Lucia */ - { 0x1F1F1, 0x1F1EE }, /* flag: Liechtenstein */ - { 0x1F1F1, 0x1F1F0 }, /* flag: Sri Lanka */ - { 0x1F1F1, 0x1F1F7 }, /* flag: Liberia */ - { 0x1F1F1, 0x1F1F8 }, /* flag: Lesotho */ - { 0x1F1F1, 0x1F1F9 }, /* flag: Lithuania */ - { 0x1F1F1, 0x1F1FA }, /* flag: Luxembourg */ - { 0x1F1F1, 0x1F1FB }, /* flag: Latvia */ - { 0x1F1F1, 0x1F1FE }, /* flag: Libya */ - { 0x1F1F2, 0x1F1E6 }, /* flag: Morocco */ - { 0x1F1F2, 0x1F1E8 }, /* flag: Monaco */ - { 0x1F1F2, 0x1F1E9 }, /* flag: Moldova */ - { 0x1F1F2, 0x1F1EA }, /* flag: Montenegro */ - { 0x1F1F2, 0x1F1EB }, /* flag: St. Martin */ - { 0x1F1F2, 0x1F1EC }, /* flag: Madagascar */ - { 0x1F1F2, 0x1F1ED }, /* flag: Marshall Islands */ - { 0x1F1F2, 0x1F1F0 }, /* flag: North Macedonia */ - { 0x1F1F2, 0x1F1F1 }, /* flag: Mali */ - { 0x1F1F2, 0x1F1F2 }, /* flag: Myanmar (Burma */ - { 0x1F1F2, 0x1F1F3 }, /* flag: Mongolia */ - { 0x1F1F2, 0x1F1F4 }, /* flag: Macao SAR China */ - { 0x1F1F2, 0x1F1F5 }, /* flag: Northern Mariana Islands */ - { 0x1F1F2, 0x1F1F6 }, /* flag: Martinique */ - { 0x1F1F2, 0x1F1F7 }, /* flag: Mauritania */ - { 0x1F1F2, 0x1F1F8 }, /* flag: Montserrat */ - { 0x1F1F2, 0x1F1F9 }, /* flag: Malta */ - { 0x1F1F2, 0x1F1FA }, /* flag: Mauritius */ - { 0x1F1F2, 0x1F1FB }, /* flag: Maldives */ - { 0x1F1F2, 0x1F1FC }, /* flag: Malawi */ - { 0x1F1F2, 0x1F1FD }, /* flag: Mexico */ - { 0x1F1F2, 0x1F1FE }, /* flag: Malaysia */ - { 0x1F1F2, 0x1F1FF }, /* flag: Mozambique */ - { 0x1F1F3, 0x1F1E6 }, /* flag: Namibia */ - { 0x1F1F3, 0x1F1E8 }, /* flag: New Caledonia */ - { 0x1F1F3, 0x1F1EA }, /* flag: Niger */ - { 0x1F1F3, 0x1F1EB }, /* flag: Norfolk Island */ - { 0x1F1F3, 0x1F1EC }, /* flag: Nigeria */ - { 0x1F1F3, 0x1F1EE }, /* flag: Nicaragua */ - { 0x1F1F3, 0x1F1F1 }, /* flag: Netherlands */ - { 0x1F1F3, 0x1F1F4 }, /* flag: Norway */ - { 0x1F1F3, 0x1F1F5 }, /* flag: Nepal */ - { 0x1F1F3, 0x1F1F7 }, /* flag: Nauru */ - { 0x1F1F3, 0x1F1FA }, /* flag: Niue */ - { 0x1F1F3, 0x1F1FF }, /* flag: New Zealand */ - { 0x1F1F4, 0x1F1F2 }, /* flag: Oman */ - { 0x1F1F5, 0x1F1E6 }, /* flag: Panama */ - { 0x1F1F5, 0x1F1EA }, /* flag: Peru */ - { 0x1F1F5, 0x1F1EB }, /* flag: French Polynesia */ - { 0x1F1F5, 0x1F1EC }, /* flag: Papua New Guinea */ - { 0x1F1F5, 0x1F1ED }, /* flag: Philippines */ - { 0x1F1F5, 0x1F1F0 }, /* flag: Pakistan */ - { 0x1F1F5, 0x1F1F1 }, /* flag: Poland */ - { 0x1F1F5, 0x1F1F2 }, /* flag: St. Pierre & Miquelon */ - { 0x1F1F5, 0x1F1F3 }, /* flag: Pitcairn Islands */ - { 0x1F1F5, 0x1F1F7 }, /* flag: Puerto Rico */ - { 0x1F1F5, 0x1F1F8 }, /* flag: Palestinian Territories */ - { 0x1F1F5, 0x1F1F9 }, /* flag: Portugal */ - { 0x1F1F5, 0x1F1FC }, /* flag: Palau */ - { 0x1F1F5, 0x1F1FE }, /* flag: Paraguay */ - { 0x1F1F6, 0x1F1E6 }, /* flag: Qatar */ - { 0x1F1F7, 0x1F1EA }, /* flag: Reunion */ - { 0x1F1F7, 0x1F1F4 }, /* flag: Romania */ - { 0x1F1F7, 0x1F1F8 }, /* flag: Serbia */ - { 0x1F1F7, 0x1F1FA }, /* flag: Russia */ - { 0x1F1F7, 0x1F1FC }, /* flag: Rwanda */ - { 0x1F1F8, 0x1F1E6 }, /* flag: Saudi Arabia */ - { 0x1F1F8, 0x1F1E7 }, /* flag: Solomon Islands */ - { 0x1F1F8, 0x1F1E8 }, /* flag: Seychelles */ - { 0x1F1F8, 0x1F1E9 }, /* flag: Sudan */ - { 0x1F1F8, 0x1F1EA }, /* flag: Sweden */ - { 0x1F1F8, 0x1F1EC }, /* flag: Singapore */ - { 0x1F1F8, 0x1F1ED }, /* flag: St. Helena */ - { 0x1F1F8, 0x1F1EE }, /* flag: Slovenia */ - { 0x1F1F8, 0x1F1EF }, /* flag: Svalbard & Jan Mayen */ - { 0x1F1F8, 0x1F1F0 }, /* flag: Slovakia */ - { 0x1F1F8, 0x1F1F1 }, /* flag: Sierra Leone */ - { 0x1F1F8, 0x1F1F2 }, /* flag: San Marino */ - { 0x1F1F8, 0x1F1F3 }, /* flag: Senegal */ - { 0x1F1F8, 0x1F1F4 }, /* flag: Somalia */ - { 0x1F1F8, 0x1F1F7 }, /* flag: Suriname */ - { 0x1F1F8, 0x1F1F8 }, /* flag: South Sudan */ - { 0x1F1F8, 0x1F1F9 }, /* flag: Sao Tome & Principe */ - { 0x1F1F8, 0x1F1FB }, /* flag: El Salvador */ - { 0x1F1F8, 0x1F1FD }, /* flag: Sint Maarten */ - { 0x1F1F8, 0x1F1FE }, /* flag: Syria */ - { 0x1F1F8, 0x1F1FF }, /* flag: Eswatini */ - { 0x1F1F9, 0x1F1E6 }, /* flag: Tristan da Cunha */ - { 0x1F1F9, 0x1F1E8 }, /* flag: Turks & Caicos Islands */ - { 0x1F1F9, 0x1F1E9 }, /* flag: Chad */ - { 0x1F1F9, 0x1F1EB }, /* flag: French Southern Territories */ - { 0x1F1F9, 0x1F1EC }, /* flag: Togo */ - { 0x1F1F9, 0x1F1ED }, /* flag: Thailand */ - { 0x1F1F9, 0x1F1EF }, /* flag: Tajikistan */ - { 0x1F1F9, 0x1F1F0 }, /* flag: Tokelau */ - { 0x1F1F9, 0x1F1F1 }, /* flag: Timor-Leste */ - { 0x1F1F9, 0x1F1F2 }, /* flag: Turkmenistan */ - { 0x1F1F9, 0x1F1F3 }, /* flag: Tunisia */ - { 0x1F1F9, 0x1F1F4 }, /* flag: Tonga */ - { 0x1F1F9, 0x1F1F7 }, /* flag: Turkey */ - { 0x1F1F9, 0x1F1F9 }, /* flag: Trinidad & Tobago */ - { 0x1F1F9, 0x1F1FB }, /* flag: Tuvalu */ - { 0x1F1F9, 0x1F1FC }, /* flag: Taiwan */ - { 0x1F1F9, 0x1F1FF }, /* flag: Tanzania */ - { 0x1F1FA, 0x1F1E6 }, /* flag: Ukraine */ - { 0x1F1FA, 0x1F1EC }, /* flag: Uganda */ - { 0x1F1FA, 0x1F1F2 }, /* flag: U.S. Outlying Islands */ - { 0x1F1FA, 0x1F1F3 }, /* flag: United Nations */ - { 0x1F1FA, 0x1F1F8 }, /* flag: United States */ - { 0x1F1FA, 0x1F1FE }, /* flag: Uruguay */ - { 0x1F1FA, 0x1F1FF }, /* flag: Uzbekistan */ - { 0x1F1FB, 0x1F1E6 }, /* flag: Vatican City */ - { 0x1F1FB, 0x1F1E8 }, /* flag: St. Vincent & Grenadines */ - { 0x1F1FB, 0x1F1EA }, /* flag: Venezuela */ - { 0x1F1FB, 0x1F1EC }, /* flag: British Virgin Islands */ - { 0x1F1FB, 0x1F1EE }, /* flag: U.S. Virgin Islands */ - { 0x1F1FB, 0x1F1F3 }, /* flag: Vietnam */ - { 0x1F1FB, 0x1F1FA }, /* flag: Vanuatu */ - { 0x1F1FC, 0x1F1EB }, /* flag: Wallis & Futuna */ - { 0x1F1FC, 0x1F1F8 }, /* flag: Samoa */ - { 0x1F1FD, 0x1F1F0 }, /* flag: Kosovo */ - { 0x1F1FE, 0x1F1EA }, /* flag: Yemen */ - { 0x1F1FE, 0x1F1F9 }, /* flag: Mayotte */ - { 0x1F1FF, 0x1F1E6 }, /* flag: South Africa */ - { 0x1F1FF, 0x1F1F2 }, /* flag: Zambia */ - { 0x1F1FF, 0x1F1FC }, /* flag: Zimbabwe */ - { 0x0261D, 0x1F3FB }, /* index pointing up: light skin tone */ - { 0x0261D, 0x1F3FC }, /* index pointing up: medium-light skin tone */ - { 0x0261D, 0x1F3FD }, /* index pointing up: medium skin tone */ - { 0x0261D, 0x1F3FE }, /* index pointing up: medium-dark skin tone */ - { 0x0261D, 0x1F3FF }, /* index pointing up: dark skin tone */ - { 0x026F9, 0x1F3FB }, /* person bouncing ball: light skin tone */ - { 0x026F9, 0x1F3FC }, /* person bouncing ball: medium-light skin tone */ - { 0x026F9, 0x1F3FD }, /* person bouncing ball: medium skin tone */ - { 0x026F9, 0x1F3FE }, /* person bouncing ball: medium-dark skin tone */ - { 0x026F9, 0x1F3FF }, /* person bouncing ball: dark skin tone */ - { 0x0270A, 0x1F3FB }, /* raised fist: light skin tone */ - { 0x0270A, 0x1F3FC }, /* raised fist: medium-light skin tone */ - { 0x0270A, 0x1F3FD }, /* raised fist: medium skin tone */ - { 0x0270A, 0x1F3FE }, /* raised fist: medium-dark skin tone */ - { 0x0270A, 0x1F3FF }, /* raised fist: dark skin tone */ - { 0x0270B, 0x1F3FB }, /* raised hand: light skin tone */ - { 0x0270B, 0x1F3FC }, /* raised hand: medium-light skin tone */ - { 0x0270B, 0x1F3FD }, /* raised hand: medium skin tone */ - { 0x0270B, 0x1F3FE }, /* raised hand: medium-dark skin tone */ - { 0x0270B, 0x1F3FF }, /* raised hand: dark skin tone */ - { 0x0270C, 0x1F3FB }, /* victory hand: light skin tone */ - { 0x0270C, 0x1F3FC }, /* victory hand: medium-light skin tone */ - { 0x0270C, 0x1F3FD }, /* victory hand: medium skin tone */ - { 0x0270C, 0x1F3FE }, /* victory hand: medium-dark skin tone */ - { 0x0270C, 0x1F3FF }, /* victory hand: dark skin tone */ - { 0x0270D, 0x1F3FB }, /* writing hand: light skin tone */ - { 0x0270D, 0x1F3FC }, /* writing hand: medium-light skin tone */ - { 0x0270D, 0x1F3FD }, /* writing hand: medium skin tone */ - { 0x0270D, 0x1F3FE }, /* writing hand: medium-dark skin tone */ - { 0x0270D, 0x1F3FF }, /* writing hand: dark skin tone */ - { 0x1F385, 0x1F3FB }, /* Santa Claus: light skin tone */ - { 0x1F385, 0x1F3FC }, /* Santa Claus: medium-light skin tone */ - { 0x1F385, 0x1F3FD }, /* Santa Claus: medium skin tone */ - { 0x1F385, 0x1F3FE }, /* Santa Claus: medium-dark skin tone */ - { 0x1F385, 0x1F3FF }, /* Santa Claus: dark skin tone */ - { 0x1F3C2, 0x1F3FB }, /* snowboarder: light skin tone */ - { 0x1F3C2, 0x1F3FC }, /* snowboarder: medium-light skin tone */ - { 0x1F3C2, 0x1F3FD }, /* snowboarder: medium skin tone */ - { 0x1F3C2, 0x1F3FE }, /* snowboarder: medium-dark skin tone */ - { 0x1F3C2, 0x1F3FF }, /* snowboarder: dark skin tone */ - { 0x1F3C3, 0x1F3FB }, /* person running: light skin tone */ - { 0x1F3C3, 0x1F3FC }, /* person running: medium-light skin tone */ - { 0x1F3C3, 0x1F3FD }, /* person running: medium skin tone */ - { 0x1F3C3, 0x1F3FE }, /* person running: medium-dark skin tone */ - { 0x1F3C3, 0x1F3FF }, /* person running: dark skin tone */ - { 0x1F3C4, 0x1F3FB }, /* person surfing: light skin tone */ - { 0x1F3C4, 0x1F3FC }, /* person surfing: medium-light skin tone */ - { 0x1F3C4, 0x1F3FD }, /* person surfing: medium skin tone */ - { 0x1F3C4, 0x1F3FE }, /* person surfing: medium-dark skin tone */ - { 0x1F3C4, 0x1F3FF }, /* person surfing: dark skin tone */ - { 0x1F3C7, 0x1F3FB }, /* horse racing: light skin tone */ - { 0x1F3C7, 0x1F3FC }, /* horse racing: medium-light skin tone */ - { 0x1F3C7, 0x1F3FD }, /* horse racing: medium skin tone */ - { 0x1F3C7, 0x1F3FE }, /* horse racing: medium-dark skin tone */ - { 0x1F3C7, 0x1F3FF }, /* horse racing: dark skin tone */ - { 0x1F3CA, 0x1F3FB }, /* person swimming: light skin tone */ - { 0x1F3CA, 0x1F3FC }, /* person swimming: medium-light skin tone */ - { 0x1F3CA, 0x1F3FD }, /* person swimming: medium skin tone */ - { 0x1F3CA, 0x1F3FE }, /* person swimming: medium-dark skin tone */ - { 0x1F3CA, 0x1F3FF }, /* person swimming: dark skin tone */ - { 0x1F3CB, 0x1F3FB }, /* person lifting weights: light skin tone */ - { 0x1F3CB, 0x1F3FC }, /* person lifting weights: medium-light skin tone */ - { 0x1F3CB, 0x1F3FD }, /* person lifting weights: medium skin tone */ - { 0x1F3CB, 0x1F3FE }, /* person lifting weights: medium-dark skin tone */ - { 0x1F3CB, 0x1F3FF }, /* person lifting weights: dark skin tone */ - { 0x1F3CC, 0x1F3FB }, /* person golfing: light skin tone */ - { 0x1F3CC, 0x1F3FC }, /* person golfing: medium-light skin tone */ - { 0x1F3CC, 0x1F3FD }, /* person golfing: medium skin tone */ - { 0x1F3CC, 0x1F3FE }, /* person golfing: medium-dark skin tone */ - { 0x1F3CC, 0x1F3FF }, /* person golfing: dark skin tone */ - { 0x1F442, 0x1F3FB }, /* ear: light skin tone */ - { 0x1F442, 0x1F3FC }, /* ear: medium-light skin tone */ - { 0x1F442, 0x1F3FD }, /* ear: medium skin tone */ - { 0x1F442, 0x1F3FE }, /* ear: medium-dark skin tone */ - { 0x1F442, 0x1F3FF }, /* ear: dark skin tone */ - { 0x1F443, 0x1F3FB }, /* nose: light skin tone */ - { 0x1F443, 0x1F3FC }, /* nose: medium-light skin tone */ - { 0x1F443, 0x1F3FD }, /* nose: medium skin tone */ - { 0x1F443, 0x1F3FE }, /* nose: medium-dark skin tone */ - { 0x1F443, 0x1F3FF }, /* nose: dark skin tone */ - { 0x1F446, 0x1F3FB }, /* backhand index pointing up: light skin tone */ - { 0x1F446, 0x1F3FC }, /* backhand index pointing up: medium-light skin tone */ - { 0x1F446, 0x1F3FD }, /* backhand index pointing up: medium skin tone */ - { 0x1F446, 0x1F3FE }, /* backhand index pointing up: medium-dark skin tone */ - { 0x1F446, 0x1F3FF }, /* backhand index pointing up: dark skin tone */ - { 0x1F447, 0x1F3FB }, /* backhand index pointing down: light skin tone */ - { 0x1F447, 0x1F3FC }, /* backhand index pointing down: medium-light skin tone */ - { 0x1F447, 0x1F3FD }, /* backhand index pointing down: medium skin tone */ - { 0x1F447, 0x1F3FE }, /* backhand index pointing down: medium-dark skin tone */ - { 0x1F447, 0x1F3FF }, /* backhand index pointing down: dark skin tone */ - { 0x1F448, 0x1F3FB }, /* backhand index pointing left: light skin tone */ - { 0x1F448, 0x1F3FC }, /* backhand index pointing left: medium-light skin tone */ - { 0x1F448, 0x1F3FD }, /* backhand index pointing left: medium skin tone */ - { 0x1F448, 0x1F3FE }, /* backhand index pointing left: medium-dark skin tone */ - { 0x1F448, 0x1F3FF }, /* backhand index pointing left: dark skin tone */ - { 0x1F449, 0x1F3FB }, /* backhand index pointing right: light skin tone */ - { 0x1F449, 0x1F3FC }, /* backhand index pointing right: medium-light skin tone */ - { 0x1F449, 0x1F3FD }, /* backhand index pointing right: medium skin tone */ - { 0x1F449, 0x1F3FE }, /* backhand index pointing right: medium-dark skin tone */ - { 0x1F449, 0x1F3FF }, /* backhand index pointing right: dark skin tone */ - { 0x1F44A, 0x1F3FB }, /* oncoming fist: light skin tone */ - { 0x1F44A, 0x1F3FC }, /* oncoming fist: medium-light skin tone */ - { 0x1F44A, 0x1F3FD }, /* oncoming fist: medium skin tone */ - { 0x1F44A, 0x1F3FE }, /* oncoming fist: medium-dark skin tone */ - { 0x1F44A, 0x1F3FF }, /* oncoming fist: dark skin tone */ - { 0x1F44B, 0x1F3FB }, /* waving hand: light skin tone */ - { 0x1F44B, 0x1F3FC }, /* waving hand: medium-light skin tone */ - { 0x1F44B, 0x1F3FD }, /* waving hand: medium skin tone */ - { 0x1F44B, 0x1F3FE }, /* waving hand: medium-dark skin tone */ - { 0x1F44B, 0x1F3FF }, /* waving hand: dark skin tone */ - { 0x1F44C, 0x1F3FB }, /* OK hand: light skin tone */ - { 0x1F44C, 0x1F3FC }, /* OK hand: medium-light skin tone */ - { 0x1F44C, 0x1F3FD }, /* OK hand: medium skin tone */ - { 0x1F44C, 0x1F3FE }, /* OK hand: medium-dark skin tone */ - { 0x1F44C, 0x1F3FF }, /* OK hand: dark skin tone */ - { 0x1F44D, 0x1F3FB }, /* thumbs up: light skin tone */ - { 0x1F44D, 0x1F3FC }, /* thumbs up: medium-light skin tone */ - { 0x1F44D, 0x1F3FD }, /* thumbs up: medium skin tone */ - { 0x1F44D, 0x1F3FE }, /* thumbs up: medium-dark skin tone */ - { 0x1F44D, 0x1F3FF }, /* thumbs up: dark skin tone */ - { 0x1F44E, 0x1F3FB }, /* thumbs down: light skin tone */ - { 0x1F44E, 0x1F3FC }, /* thumbs down: medium-light skin tone */ - { 0x1F44E, 0x1F3FD }, /* thumbs down: medium skin tone */ - { 0x1F44E, 0x1F3FE }, /* thumbs down: medium-dark skin tone */ - { 0x1F44E, 0x1F3FF }, /* thumbs down: dark skin tone */ - { 0x1F44F, 0x1F3FB }, /* clapping hands: light skin tone */ - { 0x1F44F, 0x1F3FC }, /* clapping hands: medium-light skin tone */ - { 0x1F44F, 0x1F3FD }, /* clapping hands: medium skin tone */ - { 0x1F44F, 0x1F3FE }, /* clapping hands: medium-dark skin tone */ - { 0x1F44F, 0x1F3FF }, /* clapping hands: dark skin tone */ - { 0x1F450, 0x1F3FB }, /* open hands: light skin tone */ - { 0x1F450, 0x1F3FC }, /* open hands: medium-light skin tone */ - { 0x1F450, 0x1F3FD }, /* open hands: medium skin tone */ - { 0x1F450, 0x1F3FE }, /* open hands: medium-dark skin tone */ - { 0x1F450, 0x1F3FF }, /* open hands: dark skin tone */ - { 0x1F466, 0x1F3FB }, /* boy: light skin tone */ - { 0x1F466, 0x1F3FC }, /* boy: medium-light skin tone */ - { 0x1F466, 0x1F3FD }, /* boy: medium skin tone */ - { 0x1F466, 0x1F3FE }, /* boy: medium-dark skin tone */ - { 0x1F466, 0x1F3FF }, /* boy: dark skin tone */ - { 0x1F467, 0x1F3FB }, /* girl: light skin tone */ - { 0x1F467, 0x1F3FC }, /* girl: medium-light skin tone */ - { 0x1F467, 0x1F3FD }, /* girl: medium skin tone */ - { 0x1F467, 0x1F3FE }, /* girl: medium-dark skin tone */ - { 0x1F467, 0x1F3FF }, /* girl: dark skin tone */ - { 0x1F468, 0x1F3FB }, /* man: light skin tone */ - { 0x1F468, 0x1F3FC }, /* man: medium-light skin tone */ - { 0x1F468, 0x1F3FD }, /* man: medium skin tone */ - { 0x1F468, 0x1F3FE }, /* man: medium-dark skin tone */ - { 0x1F468, 0x1F3FF }, /* man: dark skin tone */ - { 0x1F469, 0x1F3FB }, /* woman: light skin tone */ - { 0x1F469, 0x1F3FC }, /* woman: medium-light skin tone */ - { 0x1F469, 0x1F3FD }, /* woman: medium skin tone */ - { 0x1F469, 0x1F3FE }, /* woman: medium-dark skin tone */ - { 0x1F469, 0x1F3FF }, /* woman: dark skin tone */ - { 0x1F46B, 0x1F3FB }, /* woman and man holding hands: light skin tone */ - { 0x1F46B, 0x1F3FC }, /* woman and man holding hands: medium-light skin tone */ - { 0x1F46B, 0x1F3FD }, /* woman and man holding hands: medium skin tone */ - { 0x1F46B, 0x1F3FE }, /* woman and man holding hands: medium-dark skin tone */ - { 0x1F46B, 0x1F3FF }, /* woman and man holding hands: dark skin tone */ - { 0x1F46C, 0x1F3FB }, /* men holding hands: light skin tone */ - { 0x1F46C, 0x1F3FC }, /* men holding hands: medium-light skin tone */ - { 0x1F46C, 0x1F3FD }, /* men holding hands: medium skin tone */ - { 0x1F46C, 0x1F3FE }, /* men holding hands: medium-dark skin tone */ - { 0x1F46C, 0x1F3FF }, /* men holding hands: dark skin tone */ - { 0x1F46D, 0x1F3FB }, /* women holding hands: light skin tone */ - { 0x1F46D, 0x1F3FC }, /* women holding hands: medium-light skin tone */ - { 0x1F46D, 0x1F3FD }, /* women holding hands: medium skin tone */ - { 0x1F46D, 0x1F3FE }, /* women holding hands: medium-dark skin tone */ - { 0x1F46D, 0x1F3FF }, /* women holding hands: dark skin tone */ - { 0x1F46E, 0x1F3FB }, /* police officer: light skin tone */ - { 0x1F46E, 0x1F3FC }, /* police officer: medium-light skin tone */ - { 0x1F46E, 0x1F3FD }, /* police officer: medium skin tone */ - { 0x1F46E, 0x1F3FE }, /* police officer: medium-dark skin tone */ - { 0x1F46E, 0x1F3FF }, /* police officer: dark skin tone */ - { 0x1F470, 0x1F3FB }, /* person with veil: light skin tone */ - { 0x1F470, 0x1F3FC }, /* person with veil: medium-light skin tone */ - { 0x1F470, 0x1F3FD }, /* person with veil: medium skin tone */ - { 0x1F470, 0x1F3FE }, /* person with veil: medium-dark skin tone */ - { 0x1F470, 0x1F3FF }, /* person with veil: dark skin tone */ - { 0x1F471, 0x1F3FB }, /* person: light skin tone, blond hair */ - { 0x1F471, 0x1F3FC }, /* person: medium-light skin tone, blond hair */ - { 0x1F471, 0x1F3FD }, /* person: medium skin tone, blond hair */ - { 0x1F471, 0x1F3FE }, /* person: medium-dark skin tone, blond hair */ - { 0x1F471, 0x1F3FF }, /* person: dark skin tone, blond hair */ - { 0x1F472, 0x1F3FB }, /* person with skullcap: light skin tone */ - { 0x1F472, 0x1F3FC }, /* person with skullcap: medium-light skin tone */ - { 0x1F472, 0x1F3FD }, /* person with skullcap: medium skin tone */ - { 0x1F472, 0x1F3FE }, /* person with skullcap: medium-dark skin tone */ - { 0x1F472, 0x1F3FF }, /* person with skullcap: dark skin tone */ - { 0x1F473, 0x1F3FB }, /* person wearing turban: light skin tone */ - { 0x1F473, 0x1F3FC }, /* person wearing turban: medium-light skin tone */ - { 0x1F473, 0x1F3FD }, /* person wearing turban: medium skin tone */ - { 0x1F473, 0x1F3FE }, /* person wearing turban: medium-dark skin tone */ - { 0x1F473, 0x1F3FF }, /* person wearing turban: dark skin tone */ - { 0x1F474, 0x1F3FB }, /* old man: light skin tone */ - { 0x1F474, 0x1F3FC }, /* old man: medium-light skin tone */ - { 0x1F474, 0x1F3FD }, /* old man: medium skin tone */ - { 0x1F474, 0x1F3FE }, /* old man: medium-dark skin tone */ - { 0x1F474, 0x1F3FF }, /* old man: dark skin tone */ - { 0x1F475, 0x1F3FB }, /* old woman: light skin tone */ - { 0x1F475, 0x1F3FC }, /* old woman: medium-light skin tone */ - { 0x1F475, 0x1F3FD }, /* old woman: medium skin tone */ - { 0x1F475, 0x1F3FE }, /* old woman: medium-dark skin tone */ - { 0x1F475, 0x1F3FF }, /* old woman: dark skin tone */ - { 0x1F476, 0x1F3FB }, /* baby: light skin tone */ - { 0x1F476, 0x1F3FC }, /* baby: medium-light skin tone */ - { 0x1F476, 0x1F3FD }, /* baby: medium skin tone */ - { 0x1F476, 0x1F3FE }, /* baby: medium-dark skin tone */ - { 0x1F476, 0x1F3FF }, /* baby: dark skin tone */ - { 0x1F477, 0x1F3FB }, /* construction worker: light skin tone */ - { 0x1F477, 0x1F3FC }, /* construction worker: medium-light skin tone */ - { 0x1F477, 0x1F3FD }, /* construction worker: medium skin tone */ - { 0x1F477, 0x1F3FE }, /* construction worker: medium-dark skin tone */ - { 0x1F477, 0x1F3FF }, /* construction worker: dark skin tone */ - { 0x1F478, 0x1F3FB }, /* princess: light skin tone */ - { 0x1F478, 0x1F3FC }, /* princess: medium-light skin tone */ - { 0x1F478, 0x1F3FD }, /* princess: medium skin tone */ - { 0x1F478, 0x1F3FE }, /* princess: medium-dark skin tone */ - { 0x1F478, 0x1F3FF }, /* princess: dark skin tone */ - { 0x1F47C, 0x1F3FB }, /* baby angel: light skin tone */ - { 0x1F47C, 0x1F3FC }, /* baby angel: medium-light skin tone */ - { 0x1F47C, 0x1F3FD }, /* baby angel: medium skin tone */ - { 0x1F47C, 0x1F3FE }, /* baby angel: medium-dark skin tone */ - { 0x1F47C, 0x1F3FF }, /* baby angel: dark skin tone */ - { 0x1F481, 0x1F3FB }, /* person tipping hand: light skin tone */ - { 0x1F481, 0x1F3FC }, /* person tipping hand: medium-light skin tone */ - { 0x1F481, 0x1F3FD }, /* person tipping hand: medium skin tone */ - { 0x1F481, 0x1F3FE }, /* person tipping hand: medium-dark skin tone */ - { 0x1F481, 0x1F3FF }, /* person tipping hand: dark skin tone */ - { 0x1F482, 0x1F3FB }, /* guard: light skin tone */ - { 0x1F482, 0x1F3FC }, /* guard: medium-light skin tone */ - { 0x1F482, 0x1F3FD }, /* guard: medium skin tone */ - { 0x1F482, 0x1F3FE }, /* guard: medium-dark skin tone */ - { 0x1F482, 0x1F3FF }, /* guard: dark skin tone */ - { 0x1F483, 0x1F3FB }, /* woman dancing: light skin tone */ - { 0x1F483, 0x1F3FC }, /* woman dancing: medium-light skin tone */ - { 0x1F483, 0x1F3FD }, /* woman dancing: medium skin tone */ - { 0x1F483, 0x1F3FE }, /* woman dancing: medium-dark skin tone */ - { 0x1F483, 0x1F3FF }, /* woman dancing: dark skin tone */ - { 0x1F485, 0x1F3FB }, /* nail polish: light skin tone */ - { 0x1F485, 0x1F3FC }, /* nail polish: medium-light skin tone */ - { 0x1F485, 0x1F3FD }, /* nail polish: medium skin tone */ - { 0x1F485, 0x1F3FE }, /* nail polish: medium-dark skin tone */ - { 0x1F485, 0x1F3FF }, /* nail polish: dark skin tone */ - { 0x1F486, 0x1F3FB }, /* person getting massage: light skin tone */ - { 0x1F486, 0x1F3FC }, /* person getting massage: medium-light skin tone */ - { 0x1F486, 0x1F3FD }, /* person getting massage: medium skin tone */ - { 0x1F486, 0x1F3FE }, /* person getting massage: medium-dark skin tone */ - { 0x1F486, 0x1F3FF }, /* person getting massage: dark skin tone */ - { 0x1F487, 0x1F3FB }, /* person getting haircut: light skin tone */ - { 0x1F487, 0x1F3FC }, /* person getting haircut: medium-light skin tone */ - { 0x1F487, 0x1F3FD }, /* person getting haircut: medium skin tone */ - { 0x1F487, 0x1F3FE }, /* person getting haircut: medium-dark skin tone */ - { 0x1F487, 0x1F3FF }, /* person getting haircut: dark skin tone */ - { 0x1F48F, 0x1F3FB }, /* kiss: light skin tone */ - { 0x1F48F, 0x1F3FC }, /* kiss: medium-light skin tone */ - { 0x1F48F, 0x1F3FD }, /* kiss: medium skin tone */ - { 0x1F48F, 0x1F3FE }, /* kiss: medium-dark skin tone */ - { 0x1F48F, 0x1F3FF }, /* kiss: dark skin tone */ - { 0x1F491, 0x1F3FB }, /* couple with heart: light skin tone */ - { 0x1F491, 0x1F3FC }, /* couple with heart: medium-light skin tone */ - { 0x1F491, 0x1F3FD }, /* couple with heart: medium skin tone */ - { 0x1F491, 0x1F3FE }, /* couple with heart: medium-dark skin tone */ - { 0x1F491, 0x1F3FF }, /* couple with heart: dark skin tone */ - { 0x1F4AA, 0x1F3FB }, /* flexed biceps: light skin tone */ - { 0x1F4AA, 0x1F3FC }, /* flexed biceps: medium-light skin tone */ - { 0x1F4AA, 0x1F3FD }, /* flexed biceps: medium skin tone */ - { 0x1F4AA, 0x1F3FE }, /* flexed biceps: medium-dark skin tone */ - { 0x1F4AA, 0x1F3FF }, /* flexed biceps: dark skin tone */ - { 0x1F574, 0x1F3FB }, /* person in suit levitating: light skin tone */ - { 0x1F574, 0x1F3FC }, /* person in suit levitating: medium-light skin tone */ - { 0x1F574, 0x1F3FD }, /* person in suit levitating: medium skin tone */ - { 0x1F574, 0x1F3FE }, /* person in suit levitating: medium-dark skin tone */ - { 0x1F574, 0x1F3FF }, /* person in suit levitating: dark skin tone */ - { 0x1F575, 0x1F3FB }, /* detective: light skin tone */ - { 0x1F575, 0x1F3FC }, /* detective: medium-light skin tone */ - { 0x1F575, 0x1F3FD }, /* detective: medium skin tone */ - { 0x1F575, 0x1F3FE }, /* detective: medium-dark skin tone */ - { 0x1F575, 0x1F3FF }, /* detective: dark skin tone */ - { 0x1F57A, 0x1F3FB }, /* man dancing: light skin tone */ - { 0x1F57A, 0x1F3FC }, /* man dancing: medium-light skin tone */ - { 0x1F57A, 0x1F3FD }, /* man dancing: medium skin tone */ - { 0x1F57A, 0x1F3FE }, /* man dancing: medium-dark skin tone */ - { 0x1F57A, 0x1F3FF }, /* man dancing: dark skin tone */ - { 0x1F590, 0x1F3FB }, /* hand with fingers splayed: light skin tone */ - { 0x1F590, 0x1F3FC }, /* hand with fingers splayed: medium-light skin tone */ - { 0x1F590, 0x1F3FD }, /* hand with fingers splayed: medium skin tone */ - { 0x1F590, 0x1F3FE }, /* hand with fingers splayed: medium-dark skin tone */ - { 0x1F590, 0x1F3FF }, /* hand with fingers splayed: dark skin tone */ - { 0x1F595, 0x1F3FB }, /* middle finger: light skin tone */ - { 0x1F595, 0x1F3FC }, /* middle finger: medium-light skin tone */ - { 0x1F595, 0x1F3FD }, /* middle finger: medium skin tone */ - { 0x1F595, 0x1F3FE }, /* middle finger: medium-dark skin tone */ - { 0x1F595, 0x1F3FF }, /* middle finger: dark skin tone */ - { 0x1F596, 0x1F3FB }, /* vulcan salute: light skin tone */ - { 0x1F596, 0x1F3FC }, /* vulcan salute: medium-light skin tone */ - { 0x1F596, 0x1F3FD }, /* vulcan salute: medium skin tone */ - { 0x1F596, 0x1F3FE }, /* vulcan salute: medium-dark skin tone */ - { 0x1F596, 0x1F3FF }, /* vulcan salute: dark skin tone */ - { 0x1F645, 0x1F3FB }, /* person gesturing NO: light skin tone */ - { 0x1F645, 0x1F3FC }, /* person gesturing NO: medium-light skin tone */ - { 0x1F645, 0x1F3FD }, /* person gesturing NO: medium skin tone */ - { 0x1F645, 0x1F3FE }, /* person gesturing NO: medium-dark skin tone */ - { 0x1F645, 0x1F3FF }, /* person gesturing NO: dark skin tone */ - { 0x1F646, 0x1F3FB }, /* person gesturing OK: light skin tone */ - { 0x1F646, 0x1F3FC }, /* person gesturing OK: medium-light skin tone */ - { 0x1F646, 0x1F3FD }, /* person gesturing OK: medium skin tone */ - { 0x1F646, 0x1F3FE }, /* person gesturing OK: medium-dark skin tone */ - { 0x1F646, 0x1F3FF }, /* person gesturing OK: dark skin tone */ - { 0x1F647, 0x1F3FB }, /* person bowing: light skin tone */ - { 0x1F647, 0x1F3FC }, /* person bowing: medium-light skin tone */ - { 0x1F647, 0x1F3FD }, /* person bowing: medium skin tone */ - { 0x1F647, 0x1F3FE }, /* person bowing: medium-dark skin tone */ - { 0x1F647, 0x1F3FF }, /* person bowing: dark skin tone */ - { 0x1F64B, 0x1F3FB }, /* person raising hand: light skin tone */ - { 0x1F64B, 0x1F3FC }, /* person raising hand: medium-light skin tone */ - { 0x1F64B, 0x1F3FD }, /* person raising hand: medium skin tone */ - { 0x1F64B, 0x1F3FE }, /* person raising hand: medium-dark skin tone */ - { 0x1F64B, 0x1F3FF }, /* person raising hand: dark skin tone */ - { 0x1F64C, 0x1F3FB }, /* raising hands: light skin tone */ - { 0x1F64C, 0x1F3FC }, /* raising hands: medium-light skin tone */ - { 0x1F64C, 0x1F3FD }, /* raising hands: medium skin tone */ - { 0x1F64C, 0x1F3FE }, /* raising hands: medium-dark skin tone */ - { 0x1F64C, 0x1F3FF }, /* raising hands: dark skin tone */ - { 0x1F64D, 0x1F3FB }, /* person frowning: light skin tone */ - { 0x1F64D, 0x1F3FC }, /* person frowning: medium-light skin tone */ - { 0x1F64D, 0x1F3FD }, /* person frowning: medium skin tone */ - { 0x1F64D, 0x1F3FE }, /* person frowning: medium-dark skin tone */ - { 0x1F64D, 0x1F3FF }, /* person frowning: dark skin tone */ - { 0x1F64E, 0x1F3FB }, /* person pouting: light skin tone */ - { 0x1F64E, 0x1F3FC }, /* person pouting: medium-light skin tone */ - { 0x1F64E, 0x1F3FD }, /* person pouting: medium skin tone */ - { 0x1F64E, 0x1F3FE }, /* person pouting: medium-dark skin tone */ - { 0x1F64E, 0x1F3FF }, /* person pouting: dark skin tone */ - { 0x1F64F, 0x1F3FB }, /* folded hands: light skin tone */ - { 0x1F64F, 0x1F3FC }, /* folded hands: medium-light skin tone */ - { 0x1F64F, 0x1F3FD }, /* folded hands: medium skin tone */ - { 0x1F64F, 0x1F3FE }, /* folded hands: medium-dark skin tone */ - { 0x1F64F, 0x1F3FF }, /* folded hands: dark skin tone */ - { 0x1F6A3, 0x1F3FB }, /* person rowing boat: light skin tone */ - { 0x1F6A3, 0x1F3FC }, /* person rowing boat: medium-light skin tone */ - { 0x1F6A3, 0x1F3FD }, /* person rowing boat: medium skin tone */ - { 0x1F6A3, 0x1F3FE }, /* person rowing boat: medium-dark skin tone */ - { 0x1F6A3, 0x1F3FF }, /* person rowing boat: dark skin tone */ - { 0x1F6B4, 0x1F3FB }, /* person biking: light skin tone */ - { 0x1F6B4, 0x1F3FC }, /* person biking: medium-light skin tone */ - { 0x1F6B4, 0x1F3FD }, /* person biking: medium skin tone */ - { 0x1F6B4, 0x1F3FE }, /* person biking: medium-dark skin tone */ - { 0x1F6B4, 0x1F3FF }, /* person biking: dark skin tone */ - { 0x1F6B5, 0x1F3FB }, /* person mountain biking: light skin tone */ - { 0x1F6B5, 0x1F3FC }, /* person mountain biking: medium-light skin tone */ - { 0x1F6B5, 0x1F3FD }, /* person mountain biking: medium skin tone */ - { 0x1F6B5, 0x1F3FE }, /* person mountain biking: medium-dark skin tone */ - { 0x1F6B5, 0x1F3FF }, /* person mountain biking: dark skin tone */ - { 0x1F6B6, 0x1F3FB }, /* person walking: light skin tone */ - { 0x1F6B6, 0x1F3FC }, /* person walking: medium-light skin tone */ - { 0x1F6B6, 0x1F3FD }, /* person walking: medium skin tone */ - { 0x1F6B6, 0x1F3FE }, /* person walking: medium-dark skin tone */ - { 0x1F6B6, 0x1F3FF }, /* person walking: dark skin tone */ - { 0x1F6C0, 0x1F3FB }, /* person taking bath: light skin tone */ - { 0x1F6C0, 0x1F3FC }, /* person taking bath: medium-light skin tone */ - { 0x1F6C0, 0x1F3FD }, /* person taking bath: medium skin tone */ - { 0x1F6C0, 0x1F3FE }, /* person taking bath: medium-dark skin tone */ - { 0x1F6C0, 0x1F3FF }, /* person taking bath: dark skin tone */ - { 0x1F6CC, 0x1F3FB }, /* person in bed: light skin tone */ - { 0x1F6CC, 0x1F3FC }, /* person in bed: medium-light skin tone */ - { 0x1F6CC, 0x1F3FD }, /* person in bed: medium skin tone */ - { 0x1F6CC, 0x1F3FE }, /* person in bed: medium-dark skin tone */ - { 0x1F6CC, 0x1F3FF }, /* person in bed: dark skin tone */ - { 0x1F90C, 0x1F3FB }, /* pinched fingers: light skin tone */ - { 0x1F90C, 0x1F3FC }, /* pinched fingers: medium-light skin tone */ - { 0x1F90C, 0x1F3FD }, /* pinched fingers: medium skin tone */ - { 0x1F90C, 0x1F3FE }, /* pinched fingers: medium-dark skin tone */ - { 0x1F90C, 0x1F3FF }, /* pinched fingers: dark skin tone */ - { 0x1F90F, 0x1F3FB }, /* pinching hand: light skin tone */ - { 0x1F90F, 0x1F3FC }, /* pinching hand: medium-light skin tone */ - { 0x1F90F, 0x1F3FD }, /* pinching hand: medium skin tone */ - { 0x1F90F, 0x1F3FE }, /* pinching hand: medium-dark skin tone */ - { 0x1F90F, 0x1F3FF }, /* pinching hand: dark skin tone */ - { 0x1F918, 0x1F3FB }, /* sign of the horns: light skin tone */ - { 0x1F918, 0x1F3FC }, /* sign of the horns: medium-light skin tone */ - { 0x1F918, 0x1F3FD }, /* sign of the horns: medium skin tone */ - { 0x1F918, 0x1F3FE }, /* sign of the horns: medium-dark skin tone */ - { 0x1F918, 0x1F3FF }, /* sign of the horns: dark skin tone */ - { 0x1F919, 0x1F3FB }, /* call me hand: light skin tone */ - { 0x1F919, 0x1F3FC }, /* call me hand: medium-light skin tone */ - { 0x1F919, 0x1F3FD }, /* call me hand: medium skin tone */ - { 0x1F919, 0x1F3FE }, /* call me hand: medium-dark skin tone */ - { 0x1F919, 0x1F3FF }, /* call me hand: dark skin tone */ - { 0x1F91A, 0x1F3FB }, /* raised back of hand: light skin tone */ - { 0x1F91A, 0x1F3FC }, /* raised back of hand: medium-light skin tone */ - { 0x1F91A, 0x1F3FD }, /* raised back of hand: medium skin tone */ - { 0x1F91A, 0x1F3FE }, /* raised back of hand: medium-dark skin tone */ - { 0x1F91A, 0x1F3FF }, /* raised back of hand: dark skin tone */ - { 0x1F91B, 0x1F3FB }, /* left-facing fist: light skin tone */ - { 0x1F91B, 0x1F3FC }, /* left-facing fist: medium-light skin tone */ - { 0x1F91B, 0x1F3FD }, /* left-facing fist: medium skin tone */ - { 0x1F91B, 0x1F3FE }, /* left-facing fist: medium-dark skin tone */ - { 0x1F91B, 0x1F3FF }, /* left-facing fist: dark skin tone */ - { 0x1F91C, 0x1F3FB }, /* right-facing fist: light skin tone */ - { 0x1F91C, 0x1F3FC }, /* right-facing fist: medium-light skin tone */ - { 0x1F91C, 0x1F3FD }, /* right-facing fist: medium skin tone */ - { 0x1F91C, 0x1F3FE }, /* right-facing fist: medium-dark skin tone */ - { 0x1F91C, 0x1F3FF }, /* right-facing fist: dark skin tone */ - { 0x1F91D, 0x1F3FB }, /* handshake: light skin tone */ - { 0x1F91D, 0x1F3FC }, /* handshake: medium-light skin tone */ - { 0x1F91D, 0x1F3FD }, /* handshake: medium skin tone */ - { 0x1F91D, 0x1F3FE }, /* handshake: medium-dark skin tone */ - { 0x1F91D, 0x1F3FF }, /* handshake: dark skin tone */ - { 0x1F91E, 0x1F3FB }, /* crossed fingers: light skin tone */ - { 0x1F91E, 0x1F3FC }, /* crossed fingers: medium-light skin tone */ - { 0x1F91E, 0x1F3FD }, /* crossed fingers: medium skin tone */ - { 0x1F91E, 0x1F3FE }, /* crossed fingers: medium-dark skin tone */ - { 0x1F91E, 0x1F3FF }, /* crossed fingers: dark skin tone */ - { 0x1F91F, 0x1F3FB }, /* love-you gesture: light skin tone */ - { 0x1F91F, 0x1F3FC }, /* love-you gesture: medium-light skin tone */ - { 0x1F91F, 0x1F3FD }, /* love-you gesture: medium skin tone */ - { 0x1F91F, 0x1F3FE }, /* love-you gesture: medium-dark skin tone */ - { 0x1F91F, 0x1F3FF }, /* love-you gesture: dark skin tone */ - { 0x1F926, 0x1F3FB }, /* person facepalming: light skin tone */ - { 0x1F926, 0x1F3FC }, /* person facepalming: medium-light skin tone */ - { 0x1F926, 0x1F3FD }, /* person facepalming: medium skin tone */ - { 0x1F926, 0x1F3FE }, /* person facepalming: medium-dark skin tone */ - { 0x1F926, 0x1F3FF }, /* person facepalming: dark skin tone */ - { 0x1F930, 0x1F3FB }, /* pregnant woman: light skin tone */ - { 0x1F930, 0x1F3FC }, /* pregnant woman: medium-light skin tone */ - { 0x1F930, 0x1F3FD }, /* pregnant woman: medium skin tone */ - { 0x1F930, 0x1F3FE }, /* pregnant woman: medium-dark skin tone */ - { 0x1F930, 0x1F3FF }, /* pregnant woman: dark skin tone */ - { 0x1F931, 0x1F3FB }, /* breast-feeding: light skin tone */ - { 0x1F931, 0x1F3FC }, /* breast-feeding: medium-light skin tone */ - { 0x1F931, 0x1F3FD }, /* breast-feeding: medium skin tone */ - { 0x1F931, 0x1F3FE }, /* breast-feeding: medium-dark skin tone */ - { 0x1F931, 0x1F3FF }, /* breast-feeding: dark skin tone */ - { 0x1F932, 0x1F3FB }, /* palms up together: light skin tone */ - { 0x1F932, 0x1F3FC }, /* palms up together: medium-light skin tone */ - { 0x1F932, 0x1F3FD }, /* palms up together: medium skin tone */ - { 0x1F932, 0x1F3FE }, /* palms up together: medium-dark skin tone */ - { 0x1F932, 0x1F3FF }, /* palms up together: dark skin tone */ - { 0x1F933, 0x1F3FB }, /* selfie: light skin tone */ - { 0x1F933, 0x1F3FC }, /* selfie: medium-light skin tone */ - { 0x1F933, 0x1F3FD }, /* selfie: medium skin tone */ - { 0x1F933, 0x1F3FE }, /* selfie: medium-dark skin tone */ - { 0x1F933, 0x1F3FF }, /* selfie: dark skin tone */ - { 0x1F934, 0x1F3FB }, /* prince: light skin tone */ - { 0x1F934, 0x1F3FC }, /* prince: medium-light skin tone */ - { 0x1F934, 0x1F3FD }, /* prince: medium skin tone */ - { 0x1F934, 0x1F3FE }, /* prince: medium-dark skin tone */ - { 0x1F934, 0x1F3FF }, /* prince: dark skin tone */ - { 0x1F935, 0x1F3FB }, /* person in tuxedo: light skin tone */ - { 0x1F935, 0x1F3FC }, /* person in tuxedo: medium-light skin tone */ - { 0x1F935, 0x1F3FD }, /* person in tuxedo: medium skin tone */ - { 0x1F935, 0x1F3FE }, /* person in tuxedo: medium-dark skin tone */ - { 0x1F935, 0x1F3FF }, /* person in tuxedo: dark skin tone */ - { 0x1F936, 0x1F3FB }, /* Mrs. Claus: light skin tone */ - { 0x1F936, 0x1F3FC }, /* Mrs. Claus: medium-light skin tone */ - { 0x1F936, 0x1F3FD }, /* Mrs. Claus: medium skin tone */ - { 0x1F936, 0x1F3FE }, /* Mrs. Claus: medium-dark skin tone */ - { 0x1F936, 0x1F3FF }, /* Mrs. Claus: dark skin tone */ - { 0x1F937, 0x1F3FB }, /* person shrugging: light skin tone */ - { 0x1F937, 0x1F3FC }, /* person shrugging: medium-light skin tone */ - { 0x1F937, 0x1F3FD }, /* person shrugging: medium skin tone */ - { 0x1F937, 0x1F3FE }, /* person shrugging: medium-dark skin tone */ - { 0x1F937, 0x1F3FF }, /* person shrugging: dark skin tone */ - { 0x1F938, 0x1F3FB }, /* person cartwheeling: light skin tone */ - { 0x1F938, 0x1F3FC }, /* person cartwheeling: medium-light skin tone */ - { 0x1F938, 0x1F3FD }, /* person cartwheeling: medium skin tone */ - { 0x1F938, 0x1F3FE }, /* person cartwheeling: medium-dark skin tone */ - { 0x1F938, 0x1F3FF }, /* person cartwheeling: dark skin tone */ - { 0x1F939, 0x1F3FB }, /* person juggling: light skin tone */ - { 0x1F939, 0x1F3FC }, /* person juggling: medium-light skin tone */ - { 0x1F939, 0x1F3FD }, /* person juggling: medium skin tone */ - { 0x1F939, 0x1F3FE }, /* person juggling: medium-dark skin tone */ - { 0x1F939, 0x1F3FF }, /* person juggling: dark skin tone */ - { 0x1F93D, 0x1F3FB }, /* person playing water polo: light skin tone */ - { 0x1F93D, 0x1F3FC }, /* person playing water polo: medium-light skin tone */ - { 0x1F93D, 0x1F3FD }, /* person playing water polo: medium skin tone */ - { 0x1F93D, 0x1F3FE }, /* person playing water polo: medium-dark skin tone */ - { 0x1F93D, 0x1F3FF }, /* person playing water polo: dark skin tone */ - { 0x1F93E, 0x1F3FB }, /* person playing handball: light skin tone */ - { 0x1F93E, 0x1F3FC }, /* person playing handball: medium-light skin tone */ - { 0x1F93E, 0x1F3FD }, /* person playing handball: medium skin tone */ - { 0x1F93E, 0x1F3FE }, /* person playing handball: medium-dark skin tone */ - { 0x1F93E, 0x1F3FF }, /* person playing handball: dark skin tone */ - { 0x1F977, 0x1F3FB }, /* ninja: light skin tone */ - { 0x1F977, 0x1F3FC }, /* ninja: medium-light skin tone */ - { 0x1F977, 0x1F3FD }, /* ninja: medium skin tone */ - { 0x1F977, 0x1F3FE }, /* ninja: medium-dark skin tone */ - { 0x1F977, 0x1F3FF }, /* ninja: dark skin tone */ - { 0x1F9B5, 0x1F3FB }, /* leg: light skin tone */ - { 0x1F9B5, 0x1F3FC }, /* leg: medium-light skin tone */ - { 0x1F9B5, 0x1F3FD }, /* leg: medium skin tone */ - { 0x1F9B5, 0x1F3FE }, /* leg: medium-dark skin tone */ - { 0x1F9B5, 0x1F3FF }, /* leg: dark skin tone */ - { 0x1F9B6, 0x1F3FB }, /* foot: light skin tone */ - { 0x1F9B6, 0x1F3FC }, /* foot: medium-light skin tone */ - { 0x1F9B6, 0x1F3FD }, /* foot: medium skin tone */ - { 0x1F9B6, 0x1F3FE }, /* foot: medium-dark skin tone */ - { 0x1F9B6, 0x1F3FF }, /* foot: dark skin tone */ - { 0x1F9B8, 0x1F3FB }, /* superhero: light skin tone */ - { 0x1F9B8, 0x1F3FC }, /* superhero: medium-light skin tone */ - { 0x1F9B8, 0x1F3FD }, /* superhero: medium skin tone */ - { 0x1F9B8, 0x1F3FE }, /* superhero: medium-dark skin tone */ - { 0x1F9B8, 0x1F3FF }, /* superhero: dark skin tone */ - { 0x1F9B9, 0x1F3FB }, /* supervillain: light skin tone */ - { 0x1F9B9, 0x1F3FC }, /* supervillain: medium-light skin tone */ - { 0x1F9B9, 0x1F3FD }, /* supervillain: medium skin tone */ - { 0x1F9B9, 0x1F3FE }, /* supervillain: medium-dark skin tone */ - { 0x1F9B9, 0x1F3FF }, /* supervillain: dark skin tone */ - { 0x1F9BB, 0x1F3FB }, /* ear with hearing aid: light skin tone */ - { 0x1F9BB, 0x1F3FC }, /* ear with hearing aid: medium-light skin tone */ - { 0x1F9BB, 0x1F3FD }, /* ear with hearing aid: medium skin tone */ - { 0x1F9BB, 0x1F3FE }, /* ear with hearing aid: medium-dark skin tone */ - { 0x1F9BB, 0x1F3FF }, /* ear with hearing aid: dark skin tone */ - { 0x1F9CD, 0x1F3FB }, /* person standing: light skin tone */ - { 0x1F9CD, 0x1F3FC }, /* person standing: medium-light skin tone */ - { 0x1F9CD, 0x1F3FD }, /* person standing: medium skin tone */ - { 0x1F9CD, 0x1F3FE }, /* person standing: medium-dark skin tone */ - { 0x1F9CD, 0x1F3FF }, /* person standing: dark skin tone */ - { 0x1F9CE, 0x1F3FB }, /* person kneeling: light skin tone */ - { 0x1F9CE, 0x1F3FC }, /* person kneeling: medium-light skin tone */ - { 0x1F9CE, 0x1F3FD }, /* person kneeling: medium skin tone */ - { 0x1F9CE, 0x1F3FE }, /* person kneeling: medium-dark skin tone */ - { 0x1F9CE, 0x1F3FF }, /* person kneeling: dark skin tone */ - { 0x1F9CF, 0x1F3FB }, /* deaf person: light skin tone */ - { 0x1F9CF, 0x1F3FC }, /* deaf person: medium-light skin tone */ - { 0x1F9CF, 0x1F3FD }, /* deaf person: medium skin tone */ - { 0x1F9CF, 0x1F3FE }, /* deaf person: medium-dark skin tone */ - { 0x1F9CF, 0x1F3FF }, /* deaf person: dark skin tone */ - { 0x1F9D1, 0x1F3FB }, /* person: light skin tone */ - { 0x1F9D1, 0x1F3FC }, /* person: medium-light skin tone */ - { 0x1F9D1, 0x1F3FD }, /* person: medium skin tone */ - { 0x1F9D1, 0x1F3FE }, /* person: medium-dark skin tone */ - { 0x1F9D1, 0x1F3FF }, /* person: dark skin tone */ - { 0x1F9D2, 0x1F3FB }, /* child: light skin tone */ - { 0x1F9D2, 0x1F3FC }, /* child: medium-light skin tone */ - { 0x1F9D2, 0x1F3FD }, /* child: medium skin tone */ - { 0x1F9D2, 0x1F3FE }, /* child: medium-dark skin tone */ - { 0x1F9D2, 0x1F3FF }, /* child: dark skin tone */ - { 0x1F9D3, 0x1F3FB }, /* older person: light skin tone */ - { 0x1F9D3, 0x1F3FC }, /* older person: medium-light skin tone */ - { 0x1F9D3, 0x1F3FD }, /* older person: medium skin tone */ - { 0x1F9D3, 0x1F3FE }, /* older person: medium-dark skin tone */ - { 0x1F9D3, 0x1F3FF }, /* older person: dark skin tone */ - { 0x1F9D4, 0x1F3FB }, /* person: light skin tone, beard */ - { 0x1F9D4, 0x1F3FC }, /* person: medium-light skin tone, beard */ - { 0x1F9D4, 0x1F3FD }, /* person: medium skin tone, beard */ - { 0x1F9D4, 0x1F3FE }, /* person: medium-dark skin tone, beard */ - { 0x1F9D4, 0x1F3FF }, /* person: dark skin tone, beard */ - { 0x1F9D5, 0x1F3FB }, /* woman with headscarf: light skin tone */ - { 0x1F9D5, 0x1F3FC }, /* woman with headscarf: medium-light skin tone */ - { 0x1F9D5, 0x1F3FD }, /* woman with headscarf: medium skin tone */ - { 0x1F9D5, 0x1F3FE }, /* woman with headscarf: medium-dark skin tone */ - { 0x1F9D5, 0x1F3FF }, /* woman with headscarf: dark skin tone */ - { 0x1F9D6, 0x1F3FB }, /* person in steamy room: light skin tone */ - { 0x1F9D6, 0x1F3FC }, /* person in steamy room: medium-light skin tone */ - { 0x1F9D6, 0x1F3FD }, /* person in steamy room: medium skin tone */ - { 0x1F9D6, 0x1F3FE }, /* person in steamy room: medium-dark skin tone */ - { 0x1F9D6, 0x1F3FF }, /* person in steamy room: dark skin tone */ - { 0x1F9D7, 0x1F3FB }, /* person climbing: light skin tone */ - { 0x1F9D7, 0x1F3FC }, /* person climbing: medium-light skin tone */ - { 0x1F9D7, 0x1F3FD }, /* person climbing: medium skin tone */ - { 0x1F9D7, 0x1F3FE }, /* person climbing: medium-dark skin tone */ - { 0x1F9D7, 0x1F3FF }, /* person climbing: dark skin tone */ - { 0x1F9D8, 0x1F3FB }, /* person in lotus position: light skin tone */ - { 0x1F9D8, 0x1F3FC }, /* person in lotus position: medium-light skin tone */ - { 0x1F9D8, 0x1F3FD }, /* person in lotus position: medium skin tone */ - { 0x1F9D8, 0x1F3FE }, /* person in lotus position: medium-dark skin tone */ - { 0x1F9D8, 0x1F3FF }, /* person in lotus position: dark skin tone */ - { 0x1F9D9, 0x1F3FB }, /* mage: light skin tone */ - { 0x1F9D9, 0x1F3FC }, /* mage: medium-light skin tone */ - { 0x1F9D9, 0x1F3FD }, /* mage: medium skin tone */ - { 0x1F9D9, 0x1F3FE }, /* mage: medium-dark skin tone */ - { 0x1F9D9, 0x1F3FF }, /* mage: dark skin tone */ - { 0x1F9DA, 0x1F3FB }, /* fairy: light skin tone */ - { 0x1F9DA, 0x1F3FC }, /* fairy: medium-light skin tone */ - { 0x1F9DA, 0x1F3FD }, /* fairy: medium skin tone */ - { 0x1F9DA, 0x1F3FE }, /* fairy: medium-dark skin tone */ - { 0x1F9DA, 0x1F3FF }, /* fairy: dark skin tone */ - { 0x1F9DB, 0x1F3FB }, /* vampire: light skin tone */ - { 0x1F9DB, 0x1F3FC }, /* vampire: medium-light skin tone */ - { 0x1F9DB, 0x1F3FD }, /* vampire: medium skin tone */ - { 0x1F9DB, 0x1F3FE }, /* vampire: medium-dark skin tone */ - { 0x1F9DB, 0x1F3FF }, /* vampire: dark skin tone */ - { 0x1F9DC, 0x1F3FB }, /* merperson: light skin tone */ - { 0x1F9DC, 0x1F3FC }, /* merperson: medium-light skin tone */ - { 0x1F9DC, 0x1F3FD }, /* merperson: medium skin tone */ - { 0x1F9DC, 0x1F3FE }, /* merperson: medium-dark skin tone */ - { 0x1F9DC, 0x1F3FF }, /* merperson: dark skin tone */ - { 0x1F9DD, 0x1F3FB }, /* elf: light skin tone */ - { 0x1F9DD, 0x1F3FC }, /* elf: medium-light skin tone */ - { 0x1F9DD, 0x1F3FD }, /* elf: medium skin tone */ - { 0x1F9DD, 0x1F3FE }, /* elf: medium-dark skin tone */ - { 0x1F9DD, 0x1F3FF }, /* elf: dark skin tone */ - { 0x1FAC3, 0x1F3FB }, /* pregnant man: light skin tone */ - { 0x1FAC3, 0x1F3FC }, /* pregnant man: medium-light skin tone */ - { 0x1FAC3, 0x1F3FD }, /* pregnant man: medium skin tone */ - { 0x1FAC3, 0x1F3FE }, /* pregnant man: medium-dark skin tone */ - { 0x1FAC3, 0x1F3FF }, /* pregnant man: dark skin tone */ - { 0x1FAC4, 0x1F3FB }, /* pregnant person: light skin tone */ - { 0x1FAC4, 0x1F3FC }, /* pregnant person: medium-light skin tone */ - { 0x1FAC4, 0x1F3FD }, /* pregnant person: medium skin tone */ - { 0x1FAC4, 0x1F3FE }, /* pregnant person: medium-dark skin tone */ - { 0x1FAC4, 0x1F3FF }, /* pregnant person: dark skin tone */ - { 0x1FAC5, 0x1F3FB }, /* person with crown: light skin tone */ - { 0x1FAC5, 0x1F3FC }, /* person with crown: medium-light skin tone */ - { 0x1FAC5, 0x1F3FD }, /* person with crown: medium skin tone */ - { 0x1FAC5, 0x1F3FE }, /* person with crown: medium-dark skin tone */ - { 0x1FAC5, 0x1F3FF }, /* person with crown: dark skin tone */ - { 0x1FAF0, 0x1F3FB }, /* hand with index finger and thumb crossed: light skin tone */ - { 0x1FAF0, 0x1F3FC }, /* hand with index finger and thumb crossed: medium-light skin tone */ - { 0x1FAF0, 0x1F3FD }, /* hand with index finger and thumb crossed: medium skin tone */ - { 0x1FAF0, 0x1F3FE }, /* hand with index finger and thumb crossed: medium-dark skin tone */ - { 0x1FAF0, 0x1F3FF }, /* hand with index finger and thumb crossed: dark skin tone */ - { 0x1FAF1, 0x1F3FB }, /* rightwards hand: light skin tone */ - { 0x1FAF1, 0x1F3FC }, /* rightwards hand: medium-light skin tone */ - { 0x1FAF1, 0x1F3FD }, /* rightwards hand: medium skin tone */ - { 0x1FAF1, 0x1F3FE }, /* rightwards hand: medium-dark skin tone */ - { 0x1FAF1, 0x1F3FF }, /* rightwards hand: dark skin tone */ - { 0x1FAF2, 0x1F3FB }, /* leftwards hand: light skin tone */ - { 0x1FAF2, 0x1F3FC }, /* leftwards hand: medium-light skin tone */ - { 0x1FAF2, 0x1F3FD }, /* leftwards hand: medium skin tone */ - { 0x1FAF2, 0x1F3FE }, /* leftwards hand: medium-dark skin tone */ - { 0x1FAF2, 0x1F3FF }, /* leftwards hand: dark skin tone */ - { 0x1FAF3, 0x1F3FB }, /* palm down hand: light skin tone */ - { 0x1FAF3, 0x1F3FC }, /* palm down hand: medium-light skin tone */ - { 0x1FAF3, 0x1F3FD }, /* palm down hand: medium skin tone */ - { 0x1FAF3, 0x1F3FE }, /* palm down hand: medium-dark skin tone */ - { 0x1FAF3, 0x1F3FF }, /* palm down hand: dark skin tone */ - { 0x1FAF4, 0x1F3FB }, /* palm up hand: light skin tone */ - { 0x1FAF4, 0x1F3FC }, /* palm up hand: medium-light skin tone */ - { 0x1FAF4, 0x1F3FD }, /* palm up hand: medium skin tone */ - { 0x1FAF4, 0x1F3FE }, /* palm up hand: medium-dark skin tone */ - { 0x1FAF4, 0x1F3FF }, /* palm up hand: dark skin tone */ - { 0x1FAF5, 0x1F3FB }, /* index pointing at the viewer: light skin tone */ - { 0x1FAF5, 0x1F3FC }, /* index pointing at the viewer: medium-light skin tone */ - { 0x1FAF5, 0x1F3FD }, /* index pointing at the viewer: medium skin tone */ - { 0x1FAF5, 0x1F3FE }, /* index pointing at the viewer: medium-dark skin tone */ - { 0x1FAF5, 0x1F3FF }, /* index pointing at the viewer: dark skin tone */ - { 0x1FAF6, 0x1F3FB }, /* heart hands: light skin tone */ - { 0x1FAF6, 0x1F3FC }, /* heart hands: medium-light skin tone */ - { 0x1FAF6, 0x1F3FD }, /* heart hands: medium skin tone */ - { 0x1FAF6, 0x1F3FE }, /* heart hands: medium-dark skin tone */ - { 0x1FAF6, 0x1F3FF }, /* heart hands: dark skin tone */ - { 0x1FAF7, 0x1F3FB }, /* leftwards pushing hand: light skin tone */ - { 0x1FAF7, 0x1F3FC }, /* leftwards pushing hand: medium-light skin tone */ - { 0x1FAF7, 0x1F3FD }, /* leftwards pushing hand: medium skin tone */ - { 0x1FAF7, 0x1F3FE }, /* leftwards pushing hand: medium-dark skin tone */ - { 0x1FAF7, 0x1F3FF }, /* leftwards pushing hand: dark skin tone */ - { 0x1FAF8, 0x1F3FB }, /* rightwards pushing hand: light skin tone */ - { 0x1FAF8, 0x1F3FC }, /* rightwards pushing hand: medium-light skin tone */ - { 0x1FAF8, 0x1F3FD }, /* rightwards pushing hand: medium skin tone */ - { 0x1FAF8, 0x1F3FE }, /* rightwards pushing hand: medium-dark skin tone */ - { 0x1FAF8, 0x1F3FF }, /* rightwards pushing hand: dark skin tone */ +static const wchar_t utf8_modifier_table[] = { + 0x1F1E6, + 0x1F1E7, + 0x1F1E8, + 0x1F1E9, + 0x1F1EA, + 0x1F1EB, + 0x1F1EC, + 0x1F1ED, + 0x1F1EE, + 0x1F1EF, + 0x1F1F0, + 0x1F1F1, + 0x1F1F2, + 0x1F1F3, + 0x1F1F4, + 0x1F1F5, + 0x1F1F6, + 0x1F1F7, + 0x1F1F8, + 0x1F1F9, + 0x1F1FA, + 0x1F1FB, + 0x1F1FC, + 0x1F1FD, + 0x1F1FE, + 0x1F1FF, + 0x1F3FB, + 0x1F3FC, + 0x1F3FD, + 0x1F3FE, + 0x1F3FF }; -struct utf8_combined_first { - struct utf8_data first; - - struct utf8_data *second; - u_int count; - - RB_ENTRY(utf8_combined_first) entry; -}; - -static int -utf8_combined_first_cmp(struct utf8_combined_first *uf1, - struct utf8_combined_first *uf2) +/* Has this got a zero width joiner at the end? */ +int +utf8_has_zwj(const struct utf8_data *ud) { - struct utf8_data *ud1 = &uf1->first, *ud2 = &uf2->first; - - if (ud1->size < ud2->size) - return (-1); - if (ud1->size > ud2->size) - return (1); - return (memcmp(ud1->data, ud2->data, ud1->size)); -} -RB_HEAD(utf8_combined_tree, utf8_combined_first); -RB_GENERATE_STATIC(utf8_combined_tree, utf8_combined_first, entry, - utf8_combined_first_cmp); -static struct utf8_combined_tree utf8_combined_tree = - RB_INITIALIZER(utf8_combined_tree); - -static int -utf8_combined_second_cmp(const void *vp1, const void *vp2) -{ - const struct utf8_data *ud1 = vp1, *ud2 = vp2; - - if (ud1->size < ud2->size) - return (-1); - if (ud1->size > ud2->size) - return (1); - return (memcmp(ud1->data, ud2->data, ud1->size)); + if (ud->size < 3) + return (0); + return (memcmp(ud->data + ud->size - 3, "\342\200\215", 3) == 0); } -static int +/* Is this a zero width joiner? */ +int utf8_is_zwj(const struct utf8_data *ud) { - return (ud->size == 3 && memcmp(ud->data, "\342\200\215", 3) == 0); -} - -static struct utf8_data * -utf8_add_zwj(const struct utf8_data *ud) -{ - static struct utf8_data new; - - if (ud->size + 3 > UTF8_SIZE) - return (NULL); - memset(&new, 0, sizeof new); - memcpy(new.data, "\342\200\215", 3); - memcpy(new.data + 3, ud->data, ud->size); - new.size = 3 + ud->size; - new.width = ud->width; - return (&new); -} - -static struct utf8_combined_first * -utf8_find_combined_first(const struct utf8_data *first) -{ - struct utf8_combined_first uf; - - memset(&uf, 0, sizeof uf); - utf8_copy(&uf.first, first); - return (RB_FIND(utf8_combined_tree, &utf8_combined_tree, &uf)); -} - -static int -utf8_find_combined_second(struct utf8_combined_first *uf, - const struct utf8_data *second) -{ - return (bsearch(second, uf->second, uf->count, sizeof *uf->second, - utf8_combined_second_cmp) != NULL); -} - -void -utf8_build_combined(void) -{ - struct utf8_data first, second; - int mlen; - u_int i; - wchar_t wc; - struct utf8_combined_first *uf; - - for (i = 0; i < nitems(utf8_combined_table); i++) { - memset(&first, 0, sizeof first); - wc = utf8_combined_table[i].first; - mlen = wctomb(first.data, wc); - if (mlen <= 0 || mlen > UTF8_SIZE) { - log_debug("invalid combined character %05X", wc); - continue; - } - first.size = mlen; - - uf = utf8_find_combined_first(&first); - if (uf == NULL) { - uf = xcalloc(1, sizeof *uf); - utf8_copy(&uf->first, &first); - RB_INSERT(utf8_combined_tree, &utf8_combined_tree, uf); - } - - memset(&second, 0, sizeof second); - wc = utf8_combined_table[i].second; - mlen = wctomb(second.data, wc); - if (mlen <= 0 || mlen > UTF8_SIZE) { - log_debug("invalid combined character %05X", wc); - continue; - } - second.size = mlen; - - log_debug("combined character %05X+%05X = %.*s+%.*s", - utf8_combined_table[i].first, utf8_combined_table[i].second, - (int)first.size, first.data, (int)second.size, second.data); - - uf->second = xreallocarray(uf->second, uf->count + 1, - sizeof *uf->second); - utf8_copy(&uf->second[uf->count], &second); - uf->count++; - } - - RB_FOREACH(uf, utf8_combined_tree, &utf8_combined_tree) { - qsort(uf->second, uf->count, sizeof *uf->second, - utf8_combined_second_cmp); - } + if (ud->size != 3) + return (0); + return (memcmp(ud->data, "\342\200\215", 3) == 0); } +/* Is this a variation selector? */ int -utf8_try_combined(const struct utf8_data *ud, const struct utf8_data *last, - const struct utf8_data **combine, u_int *width) +utf8_is_vs(const struct utf8_data *ud) { - struct utf8_combined_first *uf; - - /* Use the incoming width by default. */ - *width = ud->width; - - /* - * If this is a zero width joiner, discard it but try to combine the - * next character. - */ - if (utf8_is_zwj(ud)) - return (UTF8_DISCARD_MAYBE_COMBINE); - - /* - * If there is a previous character to combine and it is a ZWJ, - * combine with the new character and a ZWJ. - */ - if (last != NULL && utf8_is_zwj(last)) { - *combine = utf8_add_zwj(ud); - if (*combine == NULL) - return (UTF8_DISCARD_NOW); - return (UTF8_COMBINE_NOW); - } - - /* - * If the width of this character is zero, combine onto the previous - * character. - */ - if (ud->width == 0) { - *combine = ud; - return (UTF8_COMBINE_NOW); - } - - /* - * Look up the character in the first character list, if it is missing, - * write it immediately. If it is present, write but try to combine - * later; also force the width to two. - */ - if (last == NULL) { - if (utf8_find_combined_first(ud) != NULL) { - *width = 2; - return (UTF8_WRITE_MAYBE_COMBINE); - } - return (UTF8_WRITE_NOW); - } - - /* - * This must be a potential combined character. If both first and - * second characters are on the list, combine. - */ - uf = utf8_find_combined_first(last); - if (uf != NULL && utf8_find_combined_second(uf, ud)) { - *combine = ud; - return (UTF8_COMBINE_NOW); - } - - return (UTF8_WRITE_NOW); + if (ud->size != 3) + return (0); + return (memcmp(ud->data, "\357\270\217", 3) == 0); +} + +/* Is this in the modifier table? */ +int +utf8_is_modifier(const struct utf8_data *ud) +{ + wchar_t wc; + + if (utf8_towc(ud, &wc) != UTF8_DONE) + return (0); + if (!utf8_in_table(wc, utf8_modifier_table, + nitems(utf8_modifier_table))) + return (0); + return (1); } diff --git a/usr.bin/tmux/utf8.c b/usr.bin/tmux/utf8.c index c1c56dd8a..888d0157c 100644 --- a/usr.bin/tmux/utf8.c +++ b/usr.bin/tmux/utf8.c @@ -1,4 +1,4 @@ -/* $OpenBSD: utf8.c,v 1.63 2023/09/01 14:29:11 nicm Exp $ */ +/* $OpenBSD: utf8.c,v 1.64 2023/09/15 15:49:05 nicm Exp $ */ /* * Copyright (c) 2008 Nicholas Marriott @@ -23,10 +23,174 @@ #include #include #include -#include #include "tmux.h" +static const wchar_t utf8_force_wide[] = { + 0x0261D, + 0x026F9, + 0x0270A, + 0x0270B, + 0x0270C, + 0x0270D, + 0x1F1E6, + 0x1F1E7, + 0x1F1E8, + 0x1F1E9, + 0x1F1EA, + 0x1F1EB, + 0x1F1EC, + 0x1F1ED, + 0x1F1EE, + 0x1F1EF, + 0x1F1F0, + 0x1F1F1, + 0x1F1F2, + 0x1F1F3, + 0x1F1F4, + 0x1F1F5, + 0x1F1F6, + 0x1F1F7, + 0x1F1F8, + 0x1F1F9, + 0x1F1FA, + 0x1F1FB, + 0x1F1FC, + 0x1F1FD, + 0x1F1FE, + 0x1F1FF, + 0x1F385, + 0x1F3C2, + 0x1F3C3, + 0x1F3C4, + 0x1F3C7, + 0x1F3CA, + 0x1F3CB, + 0x1F3CC, + 0x1F3FB, + 0x1F3FC, + 0x1F3FD, + 0x1F3FE, + 0x1F3FF, + 0x1F442, + 0x1F443, + 0x1F446, + 0x1F447, + 0x1F448, + 0x1F449, + 0x1F44A, + 0x1F44B, + 0x1F44C, + 0x1F44D, + 0x1F44E, + 0x1F44F, + 0x1F450, + 0x1F466, + 0x1F467, + 0x1F468, + 0x1F469, + 0x1F46B, + 0x1F46C, + 0x1F46D, + 0x1F46E, + 0x1F470, + 0x1F471, + 0x1F472, + 0x1F473, + 0x1F474, + 0x1F475, + 0x1F476, + 0x1F477, + 0x1F478, + 0x1F47C, + 0x1F481, + 0x1F482, + 0x1F483, + 0x1F485, + 0x1F486, + 0x1F487, + 0x1F48F, + 0x1F491, + 0x1F4AA, + 0x1F574, + 0x1F575, + 0x1F57A, + 0x1F590, + 0x1F595, + 0x1F596, + 0x1F645, + 0x1F646, + 0x1F647, + 0x1F64B, + 0x1F64C, + 0x1F64D, + 0x1F64E, + 0x1F64F, + 0x1F6A3, + 0x1F6B4, + 0x1F6B5, + 0x1F6B6, + 0x1F6C0, + 0x1F6CC, + 0x1F90C, + 0x1F90F, + 0x1F918, + 0x1F919, + 0x1F91A, + 0x1F91B, + 0x1F91C, + 0x1F91D, + 0x1F91E, + 0x1F91F, + 0x1F926, + 0x1F930, + 0x1F931, + 0x1F932, + 0x1F933, + 0x1F934, + 0x1F935, + 0x1F936, + 0x1F937, + 0x1F938, + 0x1F939, + 0x1F93D, + 0x1F93E, + 0x1F977, + 0x1F9B5, + 0x1F9B6, + 0x1F9B8, + 0x1F9B9, + 0x1F9BB, + 0x1F9CD, + 0x1F9CE, + 0x1F9CF, + 0x1F9D1, + 0x1F9D2, + 0x1F9D3, + 0x1F9D4, + 0x1F9D5, + 0x1F9D6, + 0x1F9D7, + 0x1F9D8, + 0x1F9D9, + 0x1F9DA, + 0x1F9DB, + 0x1F9DC, + 0x1F9DD, + 0x1FAC3, + 0x1FAC4, + 0x1FAC5, + 0x1FAF0, + 0x1FAF1, + 0x1FAF2, + 0x1FAF3, + 0x1FAF4, + 0x1FAF5, + 0x1FAF6, + 0x1FAF7, + 0x1FAF8 +}; + struct utf8_item { RB_ENTRY(utf8_item) index_entry; u_int index; @@ -123,6 +287,28 @@ utf8_put_item(const u_char *data, size_t size, u_int *index) return (0); } +static int +utf8_table_cmp(const void *vp1, const void *vp2) +{ + const wchar_t *wc1 = vp1, *wc2 = vp2; + + if (*wc1 < *wc2) + return (-1); + if (*wc1 > *wc2) + return (1); + return (0); +} + +/* Check if character in table. */ +int +utf8_in_table(wchar_t find, const wchar_t *table, u_int count) +{ + wchar_t *found; + + found = bsearch(&find, table, count, sizeof *table, utf8_table_cmp); + return (found != NULL); +} + /* Get UTF-8 character from data. */ enum utf8_state utf8_from_data(const struct utf8_data *ud, utf8_char *uc) @@ -217,16 +403,13 @@ utf8_width(struct utf8_data *ud, int *width) { wchar_t wc; - switch (mbtowc(&wc, ud->data, ud->size)) { - case -1: - log_debug("UTF-8 %.*s, mbtowc() %d", (int)ud->size, ud->data, - errno); - mbtowc(NULL, NULL, MB_CUR_MAX); - return (UTF8_ERROR); - case 0: + if (utf8_towc(ud, &wc) != UTF8_DONE) return (UTF8_ERROR); + if (utf8_in_table(wc, utf8_force_wide, nitems(utf8_force_wide))) { + *width = 2; + return (UTF8_DONE); } - log_debug("UTF-8 %.*s is %05X", (int)ud->size, ud->data, (u_int)wc); + *width = wcwidth(wc); log_debug("wcwidth(%05X) returned %d", (u_int)wc, *width); if (*width < 0) { @@ -241,6 +424,23 @@ utf8_width(struct utf8_data *ud, int *width) return (UTF8_ERROR); } +/* Convert UTF-8 character to wide character. */ +enum utf8_state +utf8_towc(const struct utf8_data *ud, wchar_t *wc) +{ + switch (mbtowc(wc, ud->data, ud->size)) { + case -1: + log_debug("UTF-8 %.*s, mbtowc() %d", (int)ud->size, ud->data, + errno); + mbtowc(NULL, NULL, MB_CUR_MAX); + return (UTF8_ERROR); + case 0: + return (UTF8_ERROR); + } + log_debug("UTF-8 %.*s is %05X", (int)ud->size, ud->data, (u_int)*wc); + return (UTF8_DONE); +} + /* * Open UTF-8 sequence. * diff --git a/usr.sbin/btrace/btrace.8 b/usr.sbin/btrace/btrace.8 index 771e3ce5f..b23346518 100644 --- a/usr.sbin/btrace/btrace.8 +++ b/usr.sbin/btrace/btrace.8 @@ -1,4 +1,4 @@ -.\" $OpenBSD: btrace.8,v 1.8 2022/12/22 19:53:24 kn Exp $ +.\" $OpenBSD: btrace.8,v 1.9 2023/09/15 10:56:46 claudio Exp $ .\" .\" Copyright (c) 2019 Martin Pieuchot .\" @@ -14,7 +14,7 @@ .\" ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF .\" OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. .\" -.Dd $Mdocdate: December 22 2022 $ +.Dd $Mdocdate: September 15 2023 $ .Dt BTRACE 8 .Os .Sh NAME @@ -24,6 +24,7 @@ .Nm btrace .Op Fl lnv .Op Fl e Ar program | Ar file +.Op Fl p Ar file .Op Ar argument ... .Sh DESCRIPTION The @@ -47,6 +48,12 @@ List all available probes. .It Fl n No action. Parse the program and then exit. +.It Fl p Ar file +Load symbols from +.Ar file +to convert +.Va ustack +addresses into function names. .It Fl v Verbose mode. Causes diff --git a/usr.sbin/btrace/btrace.c b/usr.sbin/btrace/btrace.c index 333846180..8feb0ef32 100644 --- a/usr.sbin/btrace/btrace.c +++ b/usr.sbin/btrace/btrace.c @@ -1,4 +1,4 @@ -/* $OpenBSD: btrace.c,v 1.77 2023/09/13 13:47:58 mpi Exp $ */ +/* $OpenBSD: btrace.c,v 1.78 2023/09/15 10:59:02 claudio Exp $ */ /* * Copyright (c) 2019 - 2023 Martin Pieuchot @@ -233,8 +233,8 @@ main(int argc, char *argv[]) __dead void usage(void) { - fprintf(stderr, "usage: %s [-lnv] [-e program | file] [argument ...]\n", - getprogname()); + fprintf(stderr, "usage: %s [-lnv] [-e program | file] [-p file] " + "[argument ...]\n", getprogname()); exit(1); }