summaryrefslogtreecommitdiffstats
path: root/compilerplugins/clang/duplicate-defines.cxx
blob: 74a95f2ccad73fb66995f4eb96ba63b8132b15ba (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*
 * This file is part of the LibreOffice project.
 *
 * Based on LLVM/Clang.
 *
 * This file is distributed under the University of Illinois Open Source
 * License. See LICENSE.TXT for details.
 *
 */

#include <cassert>
#include <iostream>
#include <unordered_map>

#include "plugin.hxx"

#include <clang/Frontend/CompilerInstance.h>
#include <clang/Frontend/FrontendActions.h>
#include <clang/Tooling/CommonOptionsParser.h>
#include <clang/Tooling/Refactoring.h>
#include <llvm/Support/Signals.h>

/// Finds duplicated preprocessor defines, which generally indicate that some definition
/// needs to be centralised somewhere.

namespace
{
struct Entry
{
    clang::SourceLocation m_aLoc;
};

class DuplicateDefines : public clang::PPCallbacks, public loplugin::Plugin
{
public:
    explicit DuplicateDefines(const loplugin::InstantiationData& data);
    virtual void run() override;
    void MacroDefined(const Token& MacroNameTok, const MacroDirective* MD) override;
    void MacroUndefined(const Token& MacroNameTok, const MacroDefinition& MD,
                        const MacroDirective* Undef) override;
    enum
    {
        isPPCallback = true
    };

private:
    clang::Preprocessor& m_rPP;
    std::unordered_map<std::string, Entry> m_aDefMap;
};

DuplicateDefines::DuplicateDefines(const loplugin::InstantiationData& data)
    : Plugin(data)
    , m_rPP(compiler.getPreprocessor())
{
    compiler.getPreprocessor().addPPCallbacks(std::unique_ptr<PPCallbacks>(this));
}

void DuplicateDefines::run()
{
    // nothing, only check preprocessor usage
}

void DuplicateDefines::MacroDefined(const Token& rMacroNameTok, const MacroDirective*)
{
    auto aLoc = rMacroNameTok.getLocation();
    if (ignoreLocation(aLoc))
        return;

    std::string aMacroName = m_rPP.getSpelling(rMacroNameTok);

    // some testing macro
    if (aMacroName == "RTL_STRING_CONST_FUNCTION")
        return;
    if (aMacroName == "rtl")
        return;
    // we replicate these macros in all the .hrc files
    if (aMacroName == "NC_" || aMacroName == "NNC_")
        return;
    // We define this prior to including <windows.h>:
    if (aMacroName == "WIN32_LEAN_AND_MEAN")
    {
        return;
    }
    // TODO no obvious fix for these
    if (aMacroName == "FID_SEARCH_NOW" || aMacroName == "FID_SVX_START" || aMacroName == "FN_PARAM")
        return;
    // ignore for now, requires adding too many includes to sw/
    if (aMacroName == "MM50")
        return;

    // ignore for now, we have the same define in svx and sw, but I can't remove one of them because
    // they reference strings in different resource bundles
    if (aMacroName == "STR_UNDO_COL_DELETE" || aMacroName == "STR_UNDO_ROW_DELETE"
        || aMacroName == "STR_TABLE_NUMFORMAT" || aMacroName == "STR_DELETE")
        return;

    if (m_aDefMap.emplace(aMacroName, Entry{ aLoc }).second)
    {
        return;
    }

    // Happens e.g. with macros defined in include/premac.h, which is intended to be included
    // (without include guards) multiple times:
    auto const other = m_aDefMap[aMacroName].m_aLoc;
    assert(aLoc == compiler.getSourceManager().getSpellingLoc(aLoc));
    assert(other == compiler.getSourceManager().getSpellingLoc(other));
    if ((compiler.getSourceManager().getFilename(aLoc)
         == compiler.getSourceManager().getFilename(other))
        && (compiler.getSourceManager().getSpellingLineNumber(aLoc)
            == compiler.getSourceManager().getSpellingLineNumber(other))
        && (compiler.getSourceManager().getSpellingColumnNumber(aLoc)
            == compiler.getSourceManager().getSpellingColumnNumber(other)))
    {
        return;
    }

    report(DiagnosticsEngine::Warning, "duplicate defines", aLoc);
    report(DiagnosticsEngine::Note, "previous define", other);
}

void DuplicateDefines::MacroUndefined(const Token& rMacroNameTok, const MacroDefinition& /*MD*/,
                                      const MacroDirective* /*Undef*/)
{
    auto aLoc = rMacroNameTok.getLocation();
    if (ignoreLocation(aLoc))
        return;

    std::string aMacroName = m_rPP.getSpelling(rMacroNameTok);
    m_aDefMap.erase(aMacroName);
}

loplugin::Plugin::Registration<DuplicateDefines> X("duplicatedefines", true);
}

/* vim:set shiftwidth=4 softtabstop=4 expandtab: */