# Copyright (C) 2002-2023 Free Software Foundation, Inc. # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation; either version 2, or (at your option) # any later version. # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # You should have received a copy of the GNU General Public License # along with this program. If not, see . package Automake::Location; use 5.010; use strict; use warnings FATAL => 'all'; =head1 NAME Automake::Location - a class for location tracking, with a stack of contexts =head1 SYNOPSIS use Automake::Location; # Create a new Location object my $where = new Automake::Location "foo.c:13"; # Change the location $where->set ("foo.c:14"); # Get the location (without context). # Here this should print "foo.c:14" print $where->get, "\n"; # Push a context, and change the location $where->push_context ("included from here"); $where->set ("bar.h:1"); # Print the location and the stack of context (for debugging) print $where->dump; # This should display # bar.h:1: # foo.c:14: included from here # Get the contexts (list of [$location_string, $description]) for my $pair (reverse $where->contexts) { my ($loc, $descr) = @{$pair}; ... } # Pop a context, and reset the location to the previous context. $where->pop_context; # Clone a Location. Use this when storing the state of a location # that would otherwise be modified. my $where_copy = $where->clone; # Serialize a Location object (for passing through a thread queue, # for example) my @array = $where->serialize (); # De-serialize: recreate a Location object from a queue. my $where = new Automake::Location::deserialize ($queue); =head1 DESCRIPTION C objects are used to keep track of locations in Automake, and used to produce diagnostics. A C object is made of two parts: a location string, and a stack of contexts. For instance if C is defined at line 1 in F which was included at line 14 in F, then the location string should be C<"bar.h:10"> and the context should be the pair (C<"foo.c:14">, C<"included from here">). Section I shows how to setup such a C, and access the location string or the stack of contexts. You can pass a C to C. =cut =head2 Methods =over =item C<$where = new Automake::Location ([$position])> Create and return a new Location object. =cut sub new ($;$) { my ($class, $position) = @_; my $self = { position => $position, contexts => [], }; bless $self, $class; return $self; } =item C<$location-Eset ($position)> Change the location to be C<$position>. =cut sub set ($$) { my ($self, $position) = @_; $self->{'position'} = $position; } =item C<$location-Eget> Get the location (without context). =cut sub get ($) { my ($self) = @_; return $self->{'position'}; } =item C<$location-Epush_context ($context)> Push a context to the location. =cut sub push_context ($$) { my ($self, $context) = @_; push @{$self->{'contexts'}}, [$self->get, $context]; $self->set (undef); } =item C<$where = $location-Epop_context ($context)> Pop a context, and reset the location to the previous context. =cut sub pop_context ($) { my ($self) = @_; my $pair = pop @{$self->{'contexts'}}; $self->set ($pair->[0]); return @{$pair}; } =item C<@contexts = $location-Eget_contexts> Return the array of contexts. =cut sub get_contexts ($) { my ($self) = @_; return @{$self->{'contexts'}}; } =item C<$location = $location-Eclone> Clone a Location. Use this when storing the state of a location that would otherwise be modified. =cut sub clone ($) { my ($self) = @_; my $other = new Automake::Location ($self->get); my @contexts = $self->get_contexts; for my $pair (@contexts) { push @{$other->{'contexts'}}, [@{$pair}]; } return $other; } =item C<$res = $location-Edump> Print the location and the stack of context (for debugging). =cut sub dump ($) { my ($self) = @_; my $res = ($self->get || 'INTERNAL') . ":\n"; for my $pair (reverse $self->get_contexts) { $res .= $pair->[0] || 'INTERNAL'; $res .= ": $pair->[1]\n"; } return $res; } =item C<@array = $location-Eserialize> Serialize a Location object (for passing through a thread queue, for example). =cut sub serialize ($) { my ($self) = @_; my @serial = (); push @serial, $self->get; my @contexts = $self->get_contexts; for my $pair (@contexts) { push @serial, @{$pair}; } push @serial, undef; return @serial; } =item C De-serialize: recreate a Location object from a queue. =cut sub deserialize ($) { my ($queue) = @_; my $position = $queue->dequeue (); my $self = new Automake::Location $position; while (my $position = $queue->dequeue ()) { my $context = $queue->dequeue (); push @{$self->{'contexts'}}, [$position, $context]; } return $self; } =back =head1 SEE ALSO L =head1 HISTORY Written by Alexandre Duret-Lutz EFE. =cut 1;