/*- * SPDX-License-Identifier: BSD-2-Clause-FreeBSD * * Copyright (c) 2019 Tobias Kortkamp * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ #include "config.h" #include #include #include #include #include #include #include #include #include #include #include "ast.h" #include "parser.h" #include "parser/edits.h" #include "rules.h" enum DedupAction { DEFAULT, USES, }; struct WalkerData { struct Parser *parser; }; // Prototypes static enum ASTWalkState refactor_dedup_tokens_walker(struct AST *, struct WalkerData *); enum ASTWalkState refactor_dedup_tokens_walker(struct AST *node, struct WalkerData *this) { SCOPE_MEMPOOL(pool); switch (node->type) { case AST_VARIABLE: if ((parser_settings(this->parser).behavior & PARSER_OUTPUT_EDITED) && !node->edited) { return AST_WALK_CONTINUE; } else if (skip_dedup(this->parser, node->variable.name, node->variable.modifier)) { return AST_WALK_CONTINUE; } else { struct Set *seen = mempool_set(pool, str_compare, NULL); struct Set *uses = mempool_set(pool, str_compare, NULL); enum DedupAction action = DEFAULT; char *helper = NULL; if (is_options_helper(pool, this->parser, node->variable.name, NULL, &helper, NULL)) { if (strcmp(helper, "USES") == 0 || strcmp(helper, "USES_OFF") == 0) { action = USES; } } else if (strcmp(node->variable.name, "USES") == 0) { action = USES; } struct Array *words = mempool_array(pool); ARRAY_FOREACH(node->variable.words, const char *, word) { // XXX: Handle *_DEPENDS (turn 'RUN_DEPENDS=foo>=1.5.6:misc/foo foo>0:misc/foo' // into 'RUN_DEPENDS=foo>=1.5.6:misc/foo')? switch (action) { case USES: { char *buf = str_dup(pool, word); char *args = strchr(buf, ':'); if (args) { *args = 0; } // We follow the semantics of the ports framework. // 'USES=compiler:c++11-lang compiler:c++14-lang' is // semantically equivalent to just USES=compiler:c++11-lang // since compiler_ARGS has already been set once before. // As such compiler:c++14-lang can be dropped entirely. if (!set_contains(uses, buf)) { array_append(words, word); set_add(uses, buf); set_add(seen, word); } break; } default: if (!set_contains(seen, word)) { array_append(words, word); set_add(seen, word); } break; } } if (array_len(words) < array_len(node->variable.words)) { node->edited = true; array_truncate(node->variable.words); ARRAY_JOIN(node->variable.words, words); } } break; default: break; } AST_WALK_DEFAULT(refactor_dedup_tokens_walker, node, this); return AST_WALK_CONTINUE; } PARSER_EDIT(refactor_dedup_tokens) { if (userdata != NULL) { parser_set_error(parser, PARSER_ERROR_INVALID_ARGUMENT, NULL); return; } refactor_dedup_tokens_walker(root, &(struct WalkerData){ .parser = parser, }); }