| line | stmt | bran | cond | sub | pod | time | code | 
| 1 |  |  |  |  |  |  | # Marpa::R3 is Copyright (C) 2018, Jeffrey Kegler. | 
| 2 |  |  |  |  |  |  | # | 
| 3 |  |  |  |  |  |  | # This module is free software; you can redistribute it and/or modify it | 
| 4 |  |  |  |  |  |  | # under the same terms as Perl 5.10.1. For more details, see the full text | 
| 5 |  |  |  |  |  |  | # of the licenses in the directory LICENSES. | 
| 6 |  |  |  |  |  |  | # | 
| 7 |  |  |  |  |  |  | # This program is distributed in the hope that it will be | 
| 8 |  |  |  |  |  |  | # useful, but it is provided "as is" and without any express | 
| 9 |  |  |  |  |  |  | # or implied warranties. For details, see the full text of | 
| 10 |  |  |  |  |  |  | # of the licenses in the directory LICENSES. | 
| 11 |  |  |  |  |  |  |  | 
| 12 |  |  |  |  |  |  | package Marpa::R3::Common; | 
| 13 |  |  |  |  |  |  |  | 
| 14 |  |  |  |  |  |  | # Marpa::R3 "common" methods | 
| 15 |  |  |  |  |  |  |  | 
| 16 | 101 |  |  | 101 |  | 2005 | use 5.010001; | 
|  | 101 |  |  |  |  | 388 |  | 
| 17 | 101 |  |  | 101 |  | 642 | use warnings; | 
|  | 101 |  |  |  |  | 222 |  | 
|  | 101 |  |  |  |  | 3030 |  | 
| 18 | 101 |  |  | 101 |  | 587 | use strict; | 
|  | 101 |  |  |  |  | 223 |  | 
|  | 101 |  |  |  |  | 3161 |  | 
| 19 | 101 |  |  | 101 |  | 676 | use English qw( -no_match_vars ); | 
|  | 101 |  |  |  |  | 267 |  | 
|  | 101 |  |  |  |  | 813 |  | 
| 20 |  |  |  |  |  |  |  | 
| 21 | 101 |  |  | 101 |  | 45513 | use vars qw($VERSION $STRING_VERSION); | 
|  | 101 |  |  |  |  | 242 |  | 
|  | 101 |  |  |  |  | 9084 |  | 
| 22 |  |  |  |  |  |  | $VERSION        = '4.001_052'; | 
| 23 |  |  |  |  |  |  | $STRING_VERSION = $VERSION; | 
| 24 |  |  |  |  |  |  | ## no critic(BuiltinFunctions::ProhibitStringyEval) | 
| 25 |  |  |  |  |  |  | $VERSION = eval $VERSION; | 
| 26 |  |  |  |  |  |  | ## use critic | 
| 27 |  |  |  |  |  |  |  | 
| 28 |  |  |  |  |  |  | package Marpa::R3::Internal; | 
| 29 |  |  |  |  |  |  |  | 
| 30 | 101 |  |  | 101 |  | 733 | use English qw( -no_match_vars ); | 
|  | 101 |  |  |  |  | 305 |  | 
|  | 101 |  |  |  |  | 624 |  | 
| 31 |  |  |  |  |  |  |  | 
| 32 |  |  |  |  |  |  | # Viewing methods, for debugging | 
| 33 |  |  |  |  |  |  |  | 
| 34 |  |  |  |  |  |  | my @escape_by_ord = (); | 
| 35 |  |  |  |  |  |  | $escape_by_ord[ ord q{\\} ] = q{\\\\}; | 
| 36 |  |  |  |  |  |  | $escape_by_ord[ ord eval qq{"$_"} ] = $_ | 
| 37 |  |  |  |  |  |  | for "\\t", "\\r", "\\f", "\\b", "\\a", "\\e"; | 
| 38 |  |  |  |  |  |  | $escape_by_ord[0xa] = '\\n'; | 
| 39 |  |  |  |  |  |  | $escape_by_ord[$_] //= chr $_ for 32 .. 126; | 
| 40 |  |  |  |  |  |  | $escape_by_ord[$_] //= sprintf( "\\x%02x", $_ ) for 0 .. 255; | 
| 41 |  |  |  |  |  |  |  | 
| 42 |  |  |  |  |  |  | sub Marpa::R3::escape_string { | 
| 43 | 0 |  |  | 0 | 0 | 0 | my ( $string, $length ) = @_; | 
| 44 | 0 |  |  |  |  | 0 | my $reversed = $length < 0; | 
| 45 | 0 | 0 |  |  |  | 0 | if ($reversed) { | 
| 46 | 0 |  |  |  |  | 0 | $string = reverse $string; | 
| 47 | 0 |  |  |  |  | 0 | $length = -$length; | 
| 48 |  |  |  |  |  |  | } | 
| 49 | 0 |  |  |  |  | 0 | my @escaped_chars = (); | 
| 50 | 0 |  |  |  |  | 0 | ORD: for my $ord ( map {ord} split //xms, $string ) { | 
|  | 0 |  |  |  |  | 0 |  | 
| 51 | 0 | 0 |  |  |  | 0 | last ORD if $length <= 0; | 
| 52 | 0 |  | 0 |  |  | 0 | my $escaped_char = $escape_by_ord[$ord] // sprintf( "\\x{%04x}", $ord ); | 
| 53 | 0 |  |  |  |  | 0 | $length -= length $escaped_char; | 
| 54 | 0 |  |  |  |  | 0 | push @escaped_chars, $escaped_char; | 
| 55 |  |  |  |  |  |  | } ## end for my $ord ( map {ord} split //xms, $string ) | 
| 56 | 0 | 0 |  |  |  | 0 | @escaped_chars = reverse @escaped_chars if $reversed; | 
| 57 | 0 |  |  |  |  | 0 | IX: for my $ix ( reverse 0 .. $#escaped_chars ) { | 
| 58 |  |  |  |  |  |  |  | 
| 59 |  |  |  |  |  |  | # only trailing spaces are escaped | 
| 60 | 0 | 0 |  |  |  | 0 | last IX if $escaped_chars[$ix] ne q{ }; | 
| 61 | 0 |  |  |  |  | 0 | $escaped_chars[$ix] = '\\s'; | 
| 62 |  |  |  |  |  |  | } ## end IX: for my $ix ( reverse 0 .. $#escaped_chars ) | 
| 63 | 0 |  |  |  |  | 0 | return join q{}, @escaped_chars; | 
| 64 |  |  |  |  |  |  | } ## end sub escape_string | 
| 65 |  |  |  |  |  |  |  | 
| 66 |  |  |  |  |  |  | sub Marpa::R3::flatten_hash_args { | 
| 67 | 2685 |  |  | 2685 | 0 | 6960 | my ($hash_arg_array) = @_; | 
| 68 | 2685 |  |  |  |  | 6012 | my %flat_args = (); | 
| 69 | 2685 |  |  |  |  | 4925 | for my $hash_ref (@{$hash_arg_array}) { | 
|  | 2685 |  |  |  |  | 7398 |  | 
| 70 | 3129 |  |  |  |  | 7389 | my $ref_type = ref $hash_ref; | 
| 71 | 3129 | 50 |  |  |  | 8454 | if ( not $ref_type ) { | 
| 72 | 0 |  |  |  |  | 0 | return undef, qq{"%s expects args as ref to HASH, got non-reference instead}; | 
| 73 |  |  |  |  |  |  | } ## end if ( not $ref_type ) | 
| 74 | 3129 | 50 |  |  |  | 8733 | if ( $ref_type ne 'HASH' ) { | 
| 75 | 0 |  |  |  |  | 0 | return undef, qq{"%s expects args as ref to HASH, got ref to $ref_type instead}; | 
| 76 |  |  |  |  |  |  | } ## end if ( $ref_type ne 'HASH' ) | 
| 77 | 3129 |  |  |  |  | 5115 | ARG: for my $arg_name ( keys %{$hash_ref} ) { | 
|  | 3129 |  |  |  |  | 12061 |  | 
| 78 | 3768 |  |  |  |  | 12707 | $flat_args{$arg_name} = $hash_ref->{$arg_name}; | 
| 79 |  |  |  |  |  |  | } | 
| 80 |  |  |  |  |  |  | } ## end for my $args (@hash_ref_args) | 
| 81 | 2685 |  |  |  |  | 10155 | return \%flat_args; | 
| 82 |  |  |  |  |  |  | } | 
| 83 |  |  |  |  |  |  |  | 
| 84 |  |  |  |  |  |  | sub Marpa::R3::exception { | 
| 85 | 50 |  |  | 50 | 0 | 293 | my $exception = join q{}, @_; | 
| 86 | 50 |  |  |  |  | 1282 | $exception =~ s/ \n* \z /\n/xms; | 
| 87 | 50 | 100 |  |  |  | 246 | die($exception) if $Marpa::R3::JUST_DIE; | 
| 88 | 48 |  |  |  |  | 129 | CALLER: for ( my $i = 0; 1; $i++) { | 
| 89 | 198 |  |  |  |  | 1187 | my ($package ) = caller($i); | 
| 90 | 198 | 50 |  |  |  | 583 | last CALLER if not $package; | 
| 91 | 198 | 100 |  |  |  | 698 | last CALLER if not 'Marpa::R3::' eq substr $package, 0, 11; | 
| 92 | 150 |  |  |  |  | 378 | $Carp::Internal{ $package } = 1; | 
| 93 |  |  |  |  |  |  | } | 
| 94 | 48 |  |  |  |  | 7179 | Carp::croak($exception, q{Marpa::R3 exception}); | 
| 95 |  |  |  |  |  |  | } | 
| 96 |  |  |  |  |  |  |  | 
| 97 |  |  |  |  |  |  | # Could/should this be made more efficient by caching line starts, | 
| 98 |  |  |  |  |  |  | # then binary searching? | 
| 99 |  |  |  |  |  |  | sub Marpa::R3::Internal::line_column { | 
| 100 | 10 |  |  | 10 | 0 | 25 | my ( $p_string, $pos ) = @_; | 
| 101 | 10 |  |  |  |  | 16 | state $EOL = "\n"; | 
| 102 | 10 |  |  |  |  | 39 | my $line = () = substr( ${$p_string}, 0, $pos ) =~ /$EOL/g; | 
|  | 10 |  |  |  |  | 76 |  | 
| 103 | 10 | 50 |  |  |  | 55 | my $column = $line ? $pos - $+[0] + 1 : $pos + 1; | 
| 104 | 10 |  |  |  |  | 46 | return [$line+1, $column]; | 
| 105 |  |  |  |  |  |  | } | 
| 106 |  |  |  |  |  |  |  | 
| 107 |  |  |  |  |  |  | # Returns a one-line string that is the escaped equivalent | 
| 108 |  |  |  |  |  |  | # of its arguments, and whose length is at most $max. | 
| 109 |  |  |  |  |  |  | # Returns a list of two elements: the escaped string and | 
| 110 |  |  |  |  |  |  | # a boolean indicating if it was truncated | 
| 111 |  |  |  |  |  |  | sub Marpa::R3::Internal::substr_as_line { | 
| 112 | 10 |  |  | 10 | 0 | 24 | my ( $p_string, $pos, $length, $max ) = @_; | 
| 113 | 10 |  |  |  |  | 19 | my $truncated     = 0; | 
| 114 | 10 |  |  |  |  | 19 | my $used          = 0; | 
| 115 | 10 |  |  |  |  | 21 | my @escaped_chars = (); | 
| 116 | 10 |  |  |  |  | 18 | my $trailing_ws   = 0; | 
| 117 | 10 | 50 |  |  |  | 28 | my $last_ix = $max > $length ? $pos + $length : $pos + $max; | 
| 118 | 10 |  |  |  |  | 33 | CHAR: for ( my $ix = $pos ; $ix <= $last_ix ; $ix++ ) { | 
| 119 | 244 | 50 |  |  |  | 467 | last CHAR if $used >= $max; | 
| 120 | 244 |  |  |  |  | 339 | my $char = substr ${$p_string}, $ix, 1; | 
|  | 244 |  |  |  |  | 458 |  | 
| 121 | 244 | 100 |  |  |  | 631 | $trailing_ws = $char =~ /\s/ ? $trailing_ws + 1 : 0; | 
| 122 | 244 |  |  |  |  | 374 | my $ord = ord $char; | 
| 123 | 244 |  | 33 |  |  | 538 | my $escaped_char = $escape_by_ord[$ord] // sprintf( "\\x{%04x}", $ord ); | 
| 124 |  |  |  |  |  |  |  | 
| 125 |  |  |  |  |  |  | # say STDERR "ord=$ord $escaped_char"; | 
| 126 | 244 |  |  |  |  | 350 | $used += length $escaped_char; | 
| 127 | 244 |  |  |  |  | 679 | push @escaped_chars, $escaped_char; | 
| 128 |  |  |  |  |  |  | } | 
| 129 | 10 |  |  |  |  | 27 | while ( $trailing_ws-- ) { | 
| 130 | 10 |  |  |  |  | 22 | my $ws_char = pop @escaped_chars; | 
| 131 | 10 |  |  |  |  | 30 | $used -= length $ws_char; | 
| 132 |  |  |  |  |  |  | } | 
| 133 | 10 |  |  |  |  | 27 | while ( $used > $max ) { | 
| 134 | 0 |  |  |  |  | 0 | my $excess_char = pop @escaped_chars; | 
| 135 | 0 |  |  |  |  | 0 | $used -= length $excess_char; | 
| 136 | 0 |  |  |  |  | 0 | $truncated = 1; | 
| 137 |  |  |  |  |  |  | } | 
| 138 | 10 |  |  |  |  | 67 | return ( join q{}, @escaped_chars ), $truncated; | 
| 139 |  |  |  |  |  |  | } | 
| 140 |  |  |  |  |  |  |  | 
| 141 |  |  |  |  |  |  | # Returns a two-line summary of a substring -- | 
| 142 |  |  |  |  |  |  | # a first line with descriptive information and | 
| 143 |  |  |  |  |  |  | # a one-line escaped version, indented 2 spaces | 
| 144 |  |  |  |  |  |  | sub Marpa::R3::Internal::substr_as_2lines { | 
| 145 | 10 |  |  | 10 | 0 | 30 | my ( $what, $p_string, $pos, $length, $max ) = @_; | 
| 146 | 10 |  |  |  |  | 32 | my ($escaped, $trunc) = substr_as_line( $p_string, $pos, $length, $max ); | 
| 147 | 10 |  |  |  |  | 25 | my ($line_no, $column) = @{line_column( $p_string, $pos)}; | 
|  | 10 |  |  |  |  | 27 |  | 
| 148 | 10 |  |  |  |  | 30 | my @pieces = ($what); | 
| 149 | 10 | 50 |  |  |  | 29 | push @pieces, $trunc ? 'begins' : 'is'; | 
| 150 | 10 |  |  |  |  | 37 | push @pieces, qq{at line $line_no, column $column:}; | 
| 151 | 10 |  |  |  |  | 27 | my $line1 = join q{ }, @pieces; | 
| 152 | 10 |  |  |  |  | 53 | return "$line1\n  $escaped"; | 
| 153 |  |  |  |  |  |  | } | 
| 154 |  |  |  |  |  |  |  | 
| 155 |  |  |  |  |  |  | 1; | 
| 156 |  |  |  |  |  |  |  | 
| 157 |  |  |  |  |  |  | # vim: set expandtab shiftwidth=4: |