Skip to content

Commit

Permalink
Merge pull request #1788 from pocke/Omit_unnecessary_field_from_locat…
Browse files Browse the repository at this point in the history
…ion_range

Omit unnecessary field from location range
  • Loading branch information
pocke authored Sep 6, 2024
2 parents 65f6d48 + 553543d commit 9777d42
Show file tree
Hide file tree
Showing 13 changed files with 97 additions and 55 deletions.
2 changes: 1 addition & 1 deletion ext/rbs_extension/lexer.c
Original file line number Diff line number Diff line change
Expand Up @@ -116,7 +116,7 @@ token rbsparser_next_token(lexstate *state) {
yy1:
rbs_skip(state);
#line 144 "ext/rbs_extension/lexer.re"
{ return next_token(state, pEOF); }
{ return next_eof_token(state); }
#line 121 "ext/rbs_extension/lexer.c"
yy2:
rbs_skip(state);
Expand Down
5 changes: 5 additions & 0 deletions ext/rbs_extension/lexer.h
Original file line number Diff line number Diff line change
Expand Up @@ -167,6 +167,11 @@ void skipn(lexstate *state, size_t size);
* */
token next_token(lexstate *state, enum TokenType type);

/**
* Return new token with EOF type.
* */
token next_eof_token(lexstate *state);

token rbsparser_next_token(lexstate *state);

void print_token(token tok);
Expand Down
2 changes: 1 addition & 1 deletion ext/rbs_extension/lexer.re
Original file line number Diff line number Diff line change
Expand Up @@ -141,7 +141,7 @@ token rbsparser_next_token(lexstate *state) {
skip = ([ \t]+|[\r\n]);

skip { return next_token(state, tTRIVIA); }
"\x00" { return next_token(state, pEOF); }
"\x00" { return next_eof_token(state); }
* { return next_token(state, ErrorToken); }
*/
}
16 changes: 16 additions & 0 deletions ext/rbs_extension/lexstate.c
Original file line number Diff line number Diff line change
Expand Up @@ -129,6 +129,22 @@ token next_token(lexstate *state, enum TokenType type) {
return t;
}

token next_eof_token(lexstate *state) {
if (state->current.byte_pos == RSTRING_LEN(state->string)+1) {
// End of String
token t;
t.type = pEOF;
t.range.start = state->start;
t.range.end = state->start;
state->start = state->current;

return t;
} else {
// NULL byte in the middle of the string
return next_token(state, pEOF);
}
}

void rbs_skip(lexstate *state) {
if (!state->last_char) {
peek(state);
Expand Down
66 changes: 27 additions & 39 deletions ext/rbs_extension/location.c
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,9 @@
#define RBS_LOC_REQUIRED_P(loc, i) ((loc)->children->required_p & (1 << (i)))
#define RBS_LOC_OPTIONAL_P(loc, i) (!RBS_LOC_REQUIRED_P((loc), (i)))
#define RBS_LOC_CHILDREN_SIZE(cap) (sizeof(rbs_loc_children) + sizeof(rbs_loc_entry) * ((cap) - 1))
#define NULL_LOC_RANGE_P(rg) ((rg).start == -1)

rbs_loc_range RBS_LOC_NULL_RANGE = { -1, -1 };
VALUE RBS_Location;

position rbs_loc_position(int char_pos) {
Expand All @@ -16,6 +18,11 @@ position rbs_loc_position3(int char_pos, int line, int column) {
return pos;
}

rbs_loc_range rbs_new_loc_range(range rg) {
rbs_loc_range r = { rg.start.char_pos, rg.end.char_pos };
return r;
}

static void check_children_max(unsigned short n) {
size_t max = sizeof(rbs_loc_entry_bitmap) * 8;
if (n > max) {
Expand Down Expand Up @@ -51,7 +58,7 @@ void rbs_loc_add_required_child(rbs_loc *loc, ID name, range r) {

unsigned short i = loc->children->len++;
loc->children->entries[i].name = name;
loc->children->entries[i].rg = r;
loc->children->entries[i].rg = rbs_new_loc_range(r);

loc->children->required_p |= 1 << i;
}
Expand All @@ -61,10 +68,10 @@ void rbs_loc_add_optional_child(rbs_loc *loc, ID name, range r) {

unsigned short i = loc->children->len++;
loc->children->entries[i].name = name;
loc->children->entries[i].rg = r;
loc->children->entries[i].rg = rbs_new_loc_range(r);
}

void rbs_loc_init(rbs_loc *loc, VALUE buffer, range rg) {
void rbs_loc_init(rbs_loc *loc, VALUE buffer, rbs_loc_range rg) {
loc->buffer = buffer;
loc->rg = rg;
loc->children = NULL;
Expand Down Expand Up @@ -100,7 +107,7 @@ static VALUE location_s_allocate(VALUE klass) {
rbs_loc *loc;
VALUE obj = TypedData_Make_Struct(klass, rbs_loc, &location_type, loc);

rbs_loc_init(loc, Qnil, NULL_RANGE);
rbs_loc_init(loc, Qnil, RBS_LOC_NULL_RANGE);

return obj;
}
Expand All @@ -112,8 +119,8 @@ rbs_loc *rbs_check_location(VALUE obj) {
static VALUE location_initialize(VALUE self, VALUE buffer, VALUE start_pos, VALUE end_pos) {
rbs_loc *loc = rbs_check_location(self);

position start = rbs_loc_position(FIX2INT(start_pos));
position end = rbs_loc_position(FIX2INT(end_pos));
int start = FIX2INT(start_pos);
int end = FIX2INT(end_pos);

loc->buffer = buffer;
loc->rg.start = start;
Expand Down Expand Up @@ -143,38 +150,12 @@ static VALUE location_buffer(VALUE self) {

static VALUE location_start_pos(VALUE self) {
rbs_loc *loc = rbs_check_location(self);
return INT2FIX(loc->rg.start.char_pos);
return INT2FIX(loc->rg.start);
}

static VALUE location_end_pos(VALUE self) {
rbs_loc *loc = rbs_check_location(self);
return INT2FIX(loc->rg.end.char_pos);
}

static VALUE location_start_loc(VALUE self) {
rbs_loc *loc = rbs_check_location(self);

if (loc->rg.start.line >= 0) {
VALUE pair = rb_ary_new_capa(2);
rb_ary_push(pair, INT2FIX(loc->rg.start.line));
rb_ary_push(pair, INT2FIX(loc->rg.start.column));
return pair;
} else {
return Qnil;
}
}

static VALUE location_end_loc(VALUE self) {
rbs_loc *loc = rbs_check_location(self);

if (loc->rg.end.line >= 0) {
VALUE pair = rb_ary_new_capa(2);
rb_ary_push(pair, INT2FIX(loc->rg.end.line));
rb_ary_push(pair, INT2FIX(loc->rg.end.column));
return pair;
} else {
return Qnil;
}
return INT2FIX(loc->rg.end);
}

static VALUE location_add_required_child(VALUE self, VALUE name, VALUE start, VALUE end) {
Expand Down Expand Up @@ -213,6 +194,15 @@ VALUE rbs_new_location(VALUE buffer, range rg) {
rbs_loc *loc;
VALUE obj = TypedData_Make_Struct(RBS_Location, rbs_loc, &location_type, loc);

rbs_loc_init(loc, buffer, rbs_new_loc_range(rg));

return obj;
}

static VALUE rbs_new_location_from_loc_range(VALUE buffer, rbs_loc_range rg) {
rbs_loc *loc;
VALUE obj = TypedData_Make_Struct(RBS_Location, rbs_loc, &location_type, loc);

rbs_loc_init(loc, buffer, rg);

return obj;
Expand All @@ -226,12 +216,12 @@ static VALUE location_aref(VALUE self, VALUE name) {
if (loc->children != NULL) {
for (unsigned short i = 0; i < loc->children->len; i++) {
if (loc->children->entries[i].name == id) {
range result = loc->children->entries[i].rg;
rbs_loc_range result = loc->children->entries[i].rg;

if (RBS_LOC_OPTIONAL_P(loc, i) && null_range_p(result)) {
if (RBS_LOC_OPTIONAL_P(loc, i) && NULL_LOC_RANGE_P(result)) {
return Qnil;
} else {
return rbs_new_location(loc->buffer, result);
return rbs_new_location_from_loc_range(loc->buffer, result);
}
}
}
Expand Down Expand Up @@ -294,8 +284,6 @@ void rbs__init_location(void) {
rb_define_method(RBS_Location, "buffer", location_buffer, 0);
rb_define_method(RBS_Location, "start_pos", location_start_pos, 0);
rb_define_method(RBS_Location, "end_pos", location_end_pos, 0);
rb_define_private_method(RBS_Location, "_start_loc", location_start_loc, 0);
rb_define_private_method(RBS_Location, "_end_loc", location_end_loc, 0);
rb_define_method(RBS_Location, "_add_required_child", location_add_required_child, 3);
rb_define_method(RBS_Location, "_add_optional_child", location_add_optional_child, 3);
rb_define_method(RBS_Location, "_add_optional_no_child", location_add_optional_no_child, 1);
Expand Down
9 changes: 7 additions & 2 deletions ext/rbs_extension/location.h
Original file line number Diff line number Diff line change
Expand Up @@ -9,9 +9,14 @@
* */
extern VALUE RBS_Location;

typedef struct {
int start;
int end;
} rbs_loc_range;

typedef struct {
ID name;
range rg;
rbs_loc_range rg;
} rbs_loc_entry;

typedef unsigned int rbs_loc_entry_bitmap;
Expand All @@ -27,7 +32,7 @@ typedef struct {

typedef struct {
VALUE buffer;
range rg;
rbs_loc_range rg;
rbs_loc_children *children; // NULL when no children is allocated
} rbs_loc;

Expand Down
5 changes: 5 additions & 0 deletions lib/rbs/buffer.rb
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,11 @@ def ranges
@ranges << range
offset += size
end

if !content.end_with?("\n") && content.size > 0
@ranges[-1] = @ranges[-1].begin...(@ranges[-1].end+1)
end

@ranges
end
end
Expand Down
2 changes: 1 addition & 1 deletion lib/rbs/errors.rb
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@ def detailed_message(highlight: false, **)
return msg unless location.start_line == location.end_line

indent = " " * location.start_column
marker = "^" * (location.end_column - location.start_column)
marker = "^" * ([location.end_column - location.start_column, 1].max or raise)

io = StringIO.new
io.puts msg
Expand Down
8 changes: 2 additions & 6 deletions lib/rbs/location_aux.rb
Original file line number Diff line number Diff line change
Expand Up @@ -49,15 +49,11 @@ def end_column
end

def start_loc
@start_loc ||= begin
_start_loc || buffer.pos_to_loc(start_pos)
end
@start_loc ||= buffer.pos_to_loc(start_pos)
end

def end_loc
@end_loc ||= begin
_end_loc || buffer.pos_to_loc(end_pos)
end
@end_loc ||= buffer.pos_to_loc(end_pos)
end

def range
Expand Down
3 changes: 0 additions & 3 deletions sig/location.rbs
Original file line number Diff line number Diff line change
Expand Up @@ -99,9 +99,6 @@ module RBS

private

def _start_loc: () -> Buffer::loc?
def _end_loc: () -> Buffer::loc?

def _add_required_child: (RequiredChildKeys name, Integer start_pos, Integer end_pos) -> void
def _add_optional_child: (OptionalChildKeys name, Integer start_pos, Integer end_pos) -> void
def _add_optional_no_child: (OptionalChildKeys name) -> void
Expand Down
30 changes: 30 additions & 0 deletions test/rbs/buffer_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -37,4 +37,34 @@ def test_buffer

assert_equal 8, buffer.last_position
end

def test_buffer_with_no_eol
buffer = Buffer.new(name: Pathname("foo.rbs"), content: "123\nabc")

assert_equal ["123\n", "abc"], buffer.lines
assert_equal [0...4, 4...8], buffer.ranges

assert_equal [1, 0], buffer.pos_to_loc(0)
assert_equal [1, 1], buffer.pos_to_loc(1)
assert_equal [1, 2], buffer.pos_to_loc(2)
assert_equal [1, 3], buffer.pos_to_loc(3)
assert_equal [2, 0], buffer.pos_to_loc(4)
assert_equal [2, 1], buffer.pos_to_loc(5)
assert_equal [2, 2], buffer.pos_to_loc(6)
assert_equal [2, 3], buffer.pos_to_loc(7)

assert_equal 0, buffer.loc_to_pos([1, 0])
assert_equal 1, buffer.loc_to_pos([1, 1])
assert_equal 2, buffer.loc_to_pos([1, 2])
assert_equal 3, buffer.loc_to_pos([1, 3])
assert_equal 4, buffer.loc_to_pos([2, 0])
assert_equal 5, buffer.loc_to_pos([2, 1])
assert_equal 6, buffer.loc_to_pos([2, 2])
assert_equal 7, buffer.loc_to_pos([2, 3])

assert_equal "123", buffer.content[buffer.loc_to_pos([1,0])...buffer.loc_to_pos([1,3])]
assert_equal "123\n", buffer.content[buffer.loc_to_pos([1,0])...buffer.loc_to_pos([2,0])]

assert_equal 7, buffer.last_position
end
end
2 changes: 1 addition & 1 deletion test/rbs/cli_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -912,7 +912,7 @@ def test_parse_method_type

assert_raises(SystemExit) { cli.run(['parse', '--method-type', '-e', '()']) }
assert_equal [
"-e:1:2...1:3: Syntax error: expected a token `pARROW`, token=`` (pEOF) (RBS::ParsingError)",
"-e:1:2...1:2: Syntax error: expected a token `pARROW`, token=`` (pEOF) (RBS::ParsingError)",
"",
" ()",
" ^"
Expand Down
2 changes: 1 addition & 1 deletion test/rbs/parser_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -834,6 +834,6 @@ class Foo[T < Integer] < Bar # Comment
assert_equal [:tTRIVIA, "\n", 52...53], tokens.shift.then { |t| [t[0], t[1].source, t[1].range] }
assert_equal [:kEND, 'end', 53...56], tokens.shift.then { |t| [t[0], t[1].source, t[1].range] }
assert_equal [:tTRIVIA, "\n", 56...57], tokens.shift.then { |t| [t[0], t[1].source, t[1].range] }
assert_equal [:pEOF, '', 57...58], tokens.shift.then { |t| [t[0], t[1].source, t[1].range] }
assert_equal [:pEOF, '', 57...57], tokens.shift.then { |t| [t[0], t[1].source, t[1].range] }
end
end

0 comments on commit 9777d42

Please sign in to comment.