// Copyright (c) 2013 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "tools/gn/value_extractors.h" #include #include "tools/gn/build_settings.h" #include "tools/gn/err.h" #include "tools/gn/label.h" #include "tools/gn/source_dir.h" #include "tools/gn/source_file.h" #include "tools/gn/target.h" #include "tools/gn/value.h" namespace { // Sets the error and returns false on failure. template bool ListValueExtractor(const Value& value, std::vector* dest, Err* err, const Converter& converter) { if (!value.VerifyTypeIs(Value::LIST, err)) return false; const std::vector& input_list = value.list_value(); dest->resize(input_list.size()); for (size_t i = 0; i < input_list.size(); i++) { if (!converter(input_list[i], &(*dest)[i], err)) return false; } return true; } // Like the above version but extracts to a UniqueVector and sets the error if // there are duplicates. template bool ListValueUniqueExtractor(const Value& value, UniqueVector* dest, Err* err, const Converter& converter) { if (!value.VerifyTypeIs(Value::LIST, err)) return false; const std::vector& input_list = value.list_value(); for (const auto& item : input_list) { T new_one; if (!converter(item, &new_one, err)) return false; if (!dest->push_back(new_one)) { // Already in the list, throw error. *err = Err(item, "Duplicate item in list"); size_t previous_index = dest->IndexOf(new_one); err->AppendSubErr(Err(input_list[previous_index], "This was the previous definition.")); return false; } } return true; } struct RelativeFileConverter { RelativeFileConverter(const BuildSettings* build_settings_in, const SourceDir& current_dir_in) : build_settings(build_settings_in), current_dir(current_dir_in) { } bool operator()(const Value& v, SourceFile* out, Err* err) const { *out = current_dir.ResolveRelativeFile(v, err, build_settings->root_path_utf8()); return !err->has_error(); } const BuildSettings* build_settings; const SourceDir& current_dir; }; struct LibFileConverter { LibFileConverter(const BuildSettings* build_settings_in, const SourceDir& current_dir_in) : build_settings(build_settings_in), current_dir(current_dir_in) { } bool operator()(const Value& v, LibFile* out, Err* err) const { if (!v.VerifyTypeIs(Value::STRING, err)) return false; if (v.string_value().find('/') == std::string::npos) { *out = LibFile(v.string_value()); } else { *out = LibFile(current_dir.ResolveRelativeFile( v, err, build_settings->root_path_utf8())); } return !err->has_error(); } const BuildSettings* build_settings; const SourceDir& current_dir; }; struct RelativeDirConverter { RelativeDirConverter(const BuildSettings* build_settings_in, const SourceDir& current_dir_in) : build_settings(build_settings_in), current_dir(current_dir_in) { } bool operator()(const Value& v, SourceDir* out, Err* err) const { *out = current_dir.ResolveRelativeDir(v, err, build_settings->root_path_utf8()); return true; } const BuildSettings* build_settings; const SourceDir& current_dir; }; // Fills in a label. template struct LabelResolver { LabelResolver(const SourceDir& current_dir_in, const Label& current_toolchain_in) : current_dir(current_dir_in), current_toolchain(current_toolchain_in) {} bool operator()(const Value& v, Label* out, Err* err) const { if (!v.VerifyTypeIs(Value::STRING, err)) return false; *out = Label::Resolve(current_dir, current_toolchain, v, err); return !err->has_error(); } const SourceDir& current_dir; const Label& current_toolchain; }; // Fills the label part of a LabelPtrPair, leaving the pointer null. template struct LabelPtrResolver { LabelPtrResolver(const SourceDir& current_dir_in, const Label& current_toolchain_in) : current_dir(current_dir_in), current_toolchain(current_toolchain_in) {} bool operator()(const Value& v, LabelPtrPair* out, Err* err) const { if (!v.VerifyTypeIs(Value::STRING, err)) return false; out->label = Label::Resolve(current_dir, current_toolchain, v, err); out->origin = v.origin(); return !err->has_error(); } const SourceDir& current_dir; const Label& current_toolchain; }; struct LabelPatternResolver { LabelPatternResolver(const SourceDir& current_dir_in) : current_dir(current_dir_in) { } bool operator()(const Value& v, LabelPattern* out, Err* err) const { *out = LabelPattern::GetPattern(current_dir, v, err); return !err->has_error(); } const SourceDir& current_dir; }; } // namespace bool ExtractListOfStringValues(const Value& value, std::vector* dest, Err* err) { if (!value.VerifyTypeIs(Value::LIST, err)) return false; const std::vector& input_list = value.list_value(); dest->reserve(input_list.size()); for (const auto& item : input_list) { if (!item.VerifyTypeIs(Value::STRING, err)) return false; dest->push_back(item.string_value()); } return true; } bool ExtractListOfRelativeFiles(const BuildSettings* build_settings, const Value& value, const SourceDir& current_dir, std::vector* files, Err* err) { return ListValueExtractor(value, files, err, RelativeFileConverter(build_settings, current_dir)); } bool ExtractListOfLibs(const BuildSettings* build_settings, const Value& value, const SourceDir& current_dir, std::vector* libs, Err* err) { return ListValueExtractor(value, libs, err, LibFileConverter(build_settings, current_dir)); } bool ExtractListOfRelativeDirs(const BuildSettings* build_settings, const Value& value, const SourceDir& current_dir, std::vector* dest, Err* err) { return ListValueExtractor(value, dest, err, RelativeDirConverter(build_settings, current_dir)); } bool ExtractListOfLabels(const Value& value, const SourceDir& current_dir, const Label& current_toolchain, LabelTargetVector* dest, Err* err) { return ListValueExtractor(value, dest, err, LabelPtrResolver(current_dir, current_toolchain)); } bool ExtractListOfUniqueLabels(const Value& value, const SourceDir& current_dir, const Label& current_toolchain, UniqueVector