aboutsummaryrefslogtreecommitdiffhomepage
path: root/parse_execution.cpp
diff options
context:
space:
mode:
authorGravatar ridiculousfish <corydoras@ridiculousfish.com>2014-03-01 16:04:13 -0800
committerGravatar ridiculousfish <corydoras@ridiculousfish.com>2014-03-01 16:04:13 -0800
commitd232a0f9512d0e1b575235fdf4119a63353ecedd (patch)
tree8a0b4307dac8ec187a82fad92aafa65344bc6bfd /parse_execution.cpp
parentbe33d3f2a44ad16358b614c9f47fdb8094b4e127 (diff)
Implement status -n (line number) with new parser
Diffstat (limited to 'parse_execution.cpp')
-rw-r--r--parse_execution.cpp60
1 files changed, 59 insertions, 1 deletions
diff --git a/parse_execution.cpp b/parse_execution.cpp
index 726f2920..15c7de5f 100644
--- a/parse_execution.cpp
+++ b/parse_execution.cpp
@@ -50,7 +50,7 @@ static wcstring profiling_cmd_name_for_redirectable_block(const parse_node_t &no
return result;
}
-parse_execution_context_t::parse_execution_context_t(const parse_node_tree_t &t, const wcstring &s, parser_t *p, int initial_eval_level) : tree(t), src(s), parser(p), eval_level(initial_eval_level)
+parse_execution_context_t::parse_execution_context_t(const parse_node_tree_t &t, const wcstring &s, parser_t *p, int initial_eval_level) : tree(t), src(s), parser(p), eval_level(initial_eval_level), executing_node_idx(NODE_OFFSET_INVALID), cached_lineno_offset(0), cached_lineno_count(0)
{
}
@@ -1302,6 +1302,9 @@ parse_execution_result_t parse_execution_context_t::run_1_job(const parse_node_t
/* Increment the eval_level for the duration of this command */
scoped_push<int> saved_eval_level(&eval_level, eval_level + 1);
+
+ /* Save the node index */
+ scoped_push<node_offset_t> saved_node_offset(&executing_node_idx, this->get_offset(job_node));
/* Profiling support */
long long start_time = 0, parse_time = 0, exec_time = 0;
@@ -1525,3 +1528,58 @@ parse_execution_result_t parse_execution_context_t::eval_node_at_offset(node_off
return status;
}
+
+int parse_execution_context_t::get_current_line_number()
+{
+ /* If we're not executing anything, return -1 */
+ if (this->executing_node_idx == NODE_OFFSET_INVALID)
+ {
+ return -1;
+ }
+
+ /* If for some reason we're executing a node without source, return -1 */
+ const parse_node_t &node = tree.at(this->executing_node_idx);
+ if (! node.has_source())
+ {
+ return -1;
+ }
+
+ /* Count the number of newlines, leveraging our cache */
+ const size_t offset = tree.at(this->executing_node_idx).source_start;
+ assert(offset <= src.size());
+
+ /* Easy hack to handle 0 */
+ if (offset == 0)
+ {
+ return 1;
+ }
+
+ /* We want to return (one plus) the number of newlines at offsets less than the given offset. cached_lineno_count is the number of newlines at indexes less than cached_lineno_offset. */
+ const wchar_t *str = src.c_str();
+ if (offset > cached_lineno_offset)
+ {
+ size_t i;
+ for (i = cached_lineno_offset; str[i] != L'\0' && i < offset; i++)
+ {
+ /* Add one for every newline we find in the range [cached_lineno_offset, offset) */
+ if (str[i] == L'\n')
+ {
+ cached_lineno_count++;
+ }
+ }
+ cached_lineno_offset = i; //note: i, not offset, in case offset is beyond the length of the string
+ }
+ else if (offset < cached_lineno_offset)
+ {
+ /* Subtract one for every newline we find in the range [offset, cached_lineno_offset) */
+ for (size_t i = offset; i < cached_lineno_offset; i++)
+ {
+ if (str[i] == L'\n')
+ {
+ cached_lineno_count--;
+ }
+ }
+ cached_lineno_offset = offset;
+ }
+ return cached_lineno_count + 1;
+}