diff --git a/cmd-source-file.c b/cmd-source-file.c index 86c4b138..2fe124b6 100644 --- a/cmd-source-file.c +++ b/cmd-source-file.c @@ -29,6 +29,9 @@ * Sources a configuration file. */ +#define CMD_SOURCE_FILE_DEPTH_LIMIT 50 +static u_int cmd_source_file_depth; + static enum cmd_retval cmd_source_file_exec(struct cmd *, struct cmdq_item *); const struct cmd_entry cmd_source_file_entry = { @@ -59,6 +62,16 @@ struct cmd_source_file_data { static enum cmd_retval cmd_source_file_complete_cb(struct cmdq_item *item, __unused void *data) { + struct client *c = cmdq_get_client(item); + + if (c == NULL) { + cmd_source_file_depth--; + log_debug("%s: depth now %u", __func__, cmd_source_file_depth); + } else { + c->source_file_depth--; + log_debug("%s: depth now %u", __func__, c->source_file_depth); + } + cfg_print_causes(item); return (CMD_RETURN_NORMAL); } @@ -130,6 +143,7 @@ cmd_source_file_add(struct cmd_source_file_data *cdata, const char *path) path = resolved; log_debug("%s: %s", __func__, path); + cdata->files = xreallocarray(cdata->files, cdata->nfiles + 1, sizeof *cdata->files); cdata->files[cdata->nfiles++] = xstrdup(path); @@ -148,6 +162,22 @@ cmd_source_file_exec(struct cmd *self, struct cmdq_item *item) int result; u_int i, j; + if (c == NULL) { + if (cmd_source_file_depth >= CMD_SOURCE_FILE_DEPTH_LIMIT) { + cmdq_error(item, "too many nested files"); + return (CMD_RETURN_ERROR); + } + cmd_source_file_depth++; + log_debug("%s: depth now %u", __func__, cmd_source_file_depth); + } else { + if (c->source_file_depth >= CMD_SOURCE_FILE_DEPTH_LIMIT) { + cmdq_error(item, "too many nested files"); + return (CMD_RETURN_ERROR); + } + c->source_file_depth++; + log_debug("%s: depth now %u", __func__, c->source_file_depth); + } + cdata = xcalloc(1, sizeof *cdata); cdata->item = item; diff --git a/tmux.h b/tmux.h index 1a71b328..480a22b2 100644 --- a/tmux.h +++ b/tmux.h @@ -2047,6 +2047,7 @@ struct client { struct event overlay_timer; struct client_files files; + u_int source_file_depth; u_int *clipboard_panes; u_int clipboard_npanes;