/* Data and functions related to line maps and input files. Copyright (C) 2004, 2007, 2008, 2009, 2010 Free Software Foundation, Inc. This file is part of GCC. GCC 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 3, or (at your option) any later version. GCC 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 GCC; see the file COPYING3. If not see . */ #include "config.h" #include "system.h" #include "coretypes.h" #include "intl.h" #include "input.h" #include "vec.h" #include "vecprim.h" /* Current position in real source file. */ location_t input_location; struct line_maps *line_table; /* Vectors to map a discriminator-enhanced locus to a real locus and discriminator value. */ static VEC(int,heap) *discriminator_location_locations = NULL; static VEC(int,heap) *discriminator_location_discriminators = NULL; static location_t min_discriminator_location = UNKNOWN_LOCATION; expanded_location expand_location (source_location loc) { expanded_location xloc; /* If LOC describes a location with a discriminator, extract the discriminator and map it to the real location. */ if (min_discriminator_location != UNKNOWN_LOCATION && loc >= min_discriminator_location) loc = map_discriminator_location (loc); if (loc <= BUILTINS_LOCATION) { xloc.file = loc == UNKNOWN_LOCATION ? NULL : _(""); xloc.line = 0; xloc.column = 0; xloc.sysp = 0; } else { const struct line_map *map = linemap_lookup (line_table, loc); xloc.file = map->to_file; xloc.line = SOURCE_LINE (map, loc); xloc.column = SOURCE_COLUMN (map, loc); xloc.sysp = map->sysp != 0; }; return xloc; } /* Associate the DISCRIMINATOR with LOCUS, and return a new locus. We associate discriminators with a locus by allocating location_t values beyond those assigned by libcpp. Each new value is mapped directly to a real location_t value, and separately to the discriminator. */ location_t location_with_discriminator (location_t locus, int discriminator) { static int next_discriminator_location = 0; if (min_discriminator_location == UNKNOWN_LOCATION) { min_discriminator_location = line_table->highest_location + 1; next_discriminator_location = min_discriminator_location; } VEC_safe_push (int, heap, discriminator_location_locations, (int) locus); VEC_safe_push (int, heap, discriminator_location_discriminators, discriminator); return next_discriminator_location++; } /* Return TRUE if LOCUS represents a location with a discriminator. */ bool has_discriminator (location_t locus) { return (min_discriminator_location != UNKNOWN_LOCATION && locus >= min_discriminator_location); } /* Return the real location_t value for LOCUS. */ location_t map_discriminator_location (location_t locus) { if (! has_discriminator (locus)) return locus; return (location_t) VEC_index (int, discriminator_location_locations, locus - min_discriminator_location); } /* Return the discriminator for LOCUS. */ int get_discriminator_from_locus (location_t locus) { if (! has_discriminator (locus)) return 0; return VEC_index (int, discriminator_location_discriminators, locus - min_discriminator_location); }