Home | History | Annotate | Download | only in SPIRV
      1 //===- SPIRVMDWalker.h -  SPIR-V metadata walker header file ----*- C++ -*-===//
      2 //
      3 //                     The LLVM/SPIRV Translator
      4 //
      5 // This file is distributed under the University of Illinois Open Source
      6 // License. See LICENSE.TXT for details.
      7 //
      8 // Copyright (c) 2014 Advanced Micro Devices, Inc. All rights reserved.
      9 //
     10 // Permission is hereby granted, free of charge, to any person obtaining a
     11 // copy of this software and associated documentation files (the "Software"),
     12 // to deal with the Software without restriction, including without limitation
     13 // the rights to use, copy, modify, merge, publish, distribute, sublicense,
     14 // and/or sell copies of the Software, and to permit persons to whom the
     15 // Software is furnished to do so, subject to the following conditions:
     16 //
     17 // Redistributions of source code must retain the above copyright notice,
     18 // this list of conditions and the following disclaimers.
     19 // Redistributions in binary form must reproduce the above copyright notice,
     20 // this list of conditions and the following disclaimers in the documentation
     21 // and/or other materials provided with the distribution.
     22 // Neither the names of Advanced Micro Devices, Inc., nor the names of its
     23 // contributors may be used to endorse or promote products derived from this
     24 // Software without specific prior written permission.
     25 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
     26 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
     27 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
     28 // CONTRIBUTORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
     29 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
     30 // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS WITH
     31 // THE SOFTWARE.
     32 //
     33 //===----------------------------------------------------------------------===//
     34 /// \file
     35 ///
     36 /// This file declares classes for walking SPIR-V metadata.
     37 ///
     38 //===----------------------------------------------------------------------===//
     39 
     40 #ifndef LIB_SPIRV_SPIRVMDWALKER_H_
     41 #define LIB_SPIRV_SPIRVMDWALKER_H_
     42 
     43 #include "llvm/IR/Metadata.h"
     44 #include "SPIRVInternal.h"
     45 
     46 #include <functional>
     47 using namespace llvm;
     48 
     49 namespace SPIRV {
     50 
     51 class SPIRVMDWalker {
     52 public:
     53   template<typename ParentT> struct MDWrapper;
     54 
     55   struct NamedMDWrapper {
     56     NamedMDWrapper(NamedMDNode *Named, SPIRVMDWalker& WW)
     57       :NMD(Named), W(WW), I(0), Q(true){
     58       E = Named ? Named->getNumOperands() : 0;
     59     }
     60 
     61     operator bool() const { return NMD;}
     62 
     63     bool atEnd() const { return !(NMD && I < E);}
     64 
     65     MDWrapper<NamedMDWrapper> nextOp() {
     66       if (!Q)
     67         assert(I < E && "out of bound");
     68       return MDWrapper<NamedMDWrapper>((NMD && I < E) ? NMD->getOperand(I++)
     69           : nullptr, *this, W);
     70     }
     71 
     72     NamedMDWrapper &setQuiet(bool Quiet) {
     73       Q = Quiet;
     74       return *this;
     75     }
     76 
     77     NamedMDNode *NMD;
     78     SPIRVMDWalker &W;
     79     unsigned I;
     80     unsigned E;
     81     bool Q; // Quiet
     82   };
     83 
     84   template<typename ParentT>
     85   struct MDWrapper {
     86     MDWrapper(MDNode *Node, ParentT &Parent, SPIRVMDWalker &Walker)
     87       :M(Node), P(Parent), W(Walker), I(0), Q(false){
     88       E = Node ? Node->getNumOperands() : 0;
     89     }
     90 
     91     operator bool() const { return M;}
     92 
     93     bool atEnd() const { return !(M && I < E);}
     94 
     95     template<typename T>
     96     MDWrapper &get(T &V) {
     97       if (!Q)
     98         assert(I < E && "out of bound");
     99       if (atEnd())
    100         return *this;
    101       V = mdconst::dyn_extract<ConstantInt>(M->getOperand(I++))
    102           ->getZExtValue();
    103       return *this;
    104     }
    105 
    106     MDWrapper &get(std::string &S) {
    107       if (!Q)
    108         assert (I < E && "out of bound");
    109       if (atEnd())
    110         return *this;
    111       Metadata* Op = M->getOperand(I++);
    112       if (!Op)
    113         S = "";
    114       else if (auto Str = dyn_cast<MDString>(Op))
    115         S = Str->getString().str();
    116       else
    117         S = "";
    118       return *this;
    119     }
    120 
    121     MDWrapper &get(Function *&F) {
    122       if (!Q)
    123         assert (I < E && "out of bound");
    124       if (atEnd())
    125         return *this;
    126       F = mdconst::dyn_extract<Function>(M->getOperand(I++));
    127       return *this;
    128     }
    129 
    130     MDWrapper &get(SmallVectorImpl<std::string> &SV) {
    131       if (atEnd())
    132         return *this;
    133       while (I < E) {
    134         std::string S;
    135         get(S);
    136         SV.push_back(S);
    137       }
    138       return *this;
    139     }
    140 
    141     MDWrapper<MDWrapper> nextOp() {
    142       if (!Q)
    143         assert (I < E && "out of bound");
    144       return MDWrapper<MDWrapper>((M && I < E) ?
    145           dyn_cast<MDNode>(M->getOperand(I++)) : nullptr, *this, W);
    146     }
    147 
    148     ParentT &done() {
    149       return P;
    150     }
    151 
    152     MDWrapper &setQuiet(bool Quiet) {
    153       Q = Quiet;
    154       return *this;
    155     }
    156 
    157     MDNode *M;
    158     ParentT &P;
    159     SPIRVMDWalker &W;
    160     SmallVector<Metadata *, 10> V;
    161     unsigned I;
    162     unsigned E;
    163     bool Q; // Quiet
    164   };
    165 
    166   explicit SPIRVMDWalker(Module &Mod):M(Mod), C(Mod.getContext()){}
    167 
    168   NamedMDWrapper getNamedMD(StringRef Name) {
    169     return NamedMDWrapper(M.getNamedMetadata(Name), *this);
    170   }
    171 
    172   friend struct NamedMDWrapper;
    173 private:
    174   Module& M;
    175   LLVMContext& C;
    176 };
    177 
    178 } /* namespace SPIRV */
    179 
    180 #endif /* LIB_SPIRV_SPIRVMDBUILDER_H_ */
    181