/* * Cppcheck - A tool for static C/C++ code analysis * Copyright (C) 2007-2015 Daniel Marjamäki and Cppcheck team. * * This program 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 of the License, or * (at your option) any later version. * * This program 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 this program. If not, see . */ #include "cppchecklibrarydata.h" #include #include #include const unsigned int CppcheckLibraryData::Function::Arg::ANY = ~0U; CppcheckLibraryData::CppcheckLibraryData() { } static CppcheckLibraryData::Define loadDefine(const QDomElement &defineElement) { CppcheckLibraryData::Define define; define.name = defineElement.attribute("name"); define.value = defineElement.attribute("value"); return define; } static CppcheckLibraryData::Function::Arg loadFunctionArg(const QDomElement &functionArgElement) { CppcheckLibraryData::Function::Arg arg; if (functionArgElement.attribute("nr") == "any") arg.nr = CppcheckLibraryData::Function::Arg::ANY; else arg.nr = functionArgElement.attribute("nr").toUInt(); for (QDomElement childElement = functionArgElement.firstChildElement(); !childElement.isNull(); childElement = childElement.nextSiblingElement()) { if (childElement.tagName() == "not-bool") arg.notbool = true; else if (childElement.tagName() == "not-null") arg.notnull = true; else if (childElement.tagName() == "not-uninit") arg.notuninit = true; else if (childElement.tagName() == "strz") arg.strz = true; else if (childElement.tagName() == "formatstr") arg.formatstr = true; else if (childElement.tagName() == "valid") arg.valid = childElement.text(); else if (childElement.tagName() == "minsize") { CppcheckLibraryData::Function::Arg::MinSize minsize; minsize.type = childElement.attribute("type"); minsize.arg = childElement.attribute("arg"); minsize.arg2 = childElement.attribute("arg2"); arg.minsizes.append(minsize); } } return arg; } static CppcheckLibraryData::Function loadFunction(const QDomElement &functionElement, const QStringList &comments) { CppcheckLibraryData::Function function; function.comments = comments; function.name = functionElement.attribute("name"); for (QDomElement childElement = functionElement.firstChildElement(); !childElement.isNull(); childElement = childElement.nextSiblingElement()) { if (childElement.tagName() == "noreturn") function.noreturn = (childElement.text() == "true") ? CppcheckLibraryData::Function::True : CppcheckLibraryData::Function::False; else if (childElement.tagName() == "pure") function.gccPure = true; else if (childElement.tagName() == "const") function.gccConst = true; else if (childElement.tagName() == "leak-ignore") function.leakignore = true; else if (childElement.tagName() == "use-retval") function.useretval = true; else if (childElement.tagName() == "formatstr") { function.formatstr.scan = childElement.attribute("scan"); function.formatstr.secure = childElement.attribute("secure"); } else if (childElement.tagName() == "arg") { const CppcheckLibraryData::Function::Arg fa = loadFunctionArg(childElement); function.args.append(fa); } else { int x = 123; x++; } } return function; } static CppcheckLibraryData::MemoryResource loadMemoryResource(const QDomElement &element) { CppcheckLibraryData::MemoryResource memoryresource; memoryresource.type = element.tagName(); for (QDomElement childElement = element.firstChildElement(); !childElement.isNull(); childElement = childElement.nextSiblingElement()) { if (childElement.tagName() == "alloc") { CppcheckLibraryData::MemoryResource::Alloc alloc; alloc.init = (childElement.attribute("init", "false") == "true"); alloc.name = childElement.text(); memoryresource.alloc.append(alloc); } else if (childElement.tagName() == "dealloc") memoryresource.dealloc.append(childElement.text()); else if (childElement.tagName() == "use") memoryresource.use.append(childElement.text()); } return memoryresource; } static CppcheckLibraryData::PodType loadPodType(const QDomElement &element) { CppcheckLibraryData::PodType podtype; podtype.name = element.attribute("name"); podtype.size = element.attribute("size"); podtype.sign = element.attribute("sign"); return podtype; } bool CppcheckLibraryData::open(QIODevice &file) { QDomDocument doc; if (!doc.setContent(&file)) return false; clear(); QDomElement rootElement = doc.firstChildElement("def"); QStringList comments; for (QDomNode n = rootElement.firstChild(); !n.isNull(); n = n.nextSibling()) { if (n.isComment()) comments.append(n.toComment().data()); else if (n.isElement()) { const QDomElement e = n.toElement(); if (e.tagName() == "define") defines.append(loadDefine(e)); else if (e.tagName() == "function") functions.append(loadFunction(e, comments)); else if (e.tagName() == "memory" || e.tagName() == "resource") memoryresource.append(loadMemoryResource(e)); else if (e.tagName() == "podtype") podtypes.append(loadPodType(e)); comments.clear(); } } return true; } static QDomElement FunctionElement(QDomDocument &doc, const CppcheckLibraryData::Function &function) { QDomElement functionElement = doc.createElement("function"); functionElement.setAttribute("name", function.name); if (function.useretval) functionElement.appendChild(doc.createElement("use-retval")); if (function.gccConst) functionElement.appendChild(doc.createElement("const")); if (function.gccPure) functionElement.appendChild(doc.createElement("pure")); if (function.noreturn != CppcheckLibraryData::Function::Unknown) { QDomElement e = doc.createElement("noreturn"); e.appendChild(doc.createTextNode(function.noreturn == CppcheckLibraryData::Function::True ? "true" : "false")); functionElement.appendChild(e); } if (function.leakignore) functionElement.appendChild(doc.createElement("leak-ignore")); /* if (!function.formatstr.scan.isNull()) { QDomElement e = doc.createElement("formatstr"); e.setAttribute("scan", function.formatstr.scan); if (!function.formatstr.secure.isNull()) e.setAttribute("secure", function.formatstr.secure); functionElement.appendChild(e); } */ // Argument info.. foreach(const CppcheckLibraryData::Function::Arg &arg, function.args) { if (arg.formatstr) { QDomElement e = doc.createElement("formatstr"); if (!function.formatstr.scan.isNull()) e.setAttribute("scan", function.formatstr.scan); if (!function.formatstr.secure.isNull()) e.setAttribute("secure", function.formatstr.secure); functionElement.appendChild(e); } QDomElement argElement = doc.createElement("arg"); functionElement.appendChild(argElement); if (arg.nr == CppcheckLibraryData::Function::Arg::ANY) argElement.setAttribute("nr", "any"); else argElement.setAttribute("nr", arg.nr); if (arg.formatstr) argElement.appendChild(doc.createElement("formatstr")); if (arg.notnull) argElement.appendChild(doc.createElement("not-null")); if (arg.notuninit) argElement.appendChild(doc.createElement("not-uninit")); if (arg.notbool) argElement.appendChild(doc.createElement("not-bool")); if (arg.strz) argElement.appendChild(doc.createElement("strz")); if (!arg.valid.isEmpty()) { QDomElement e = doc.createElement("valid"); e.appendChild(doc.createTextNode(arg.valid)); argElement.appendChild(e); } if (!arg.minsizes.isEmpty()) { foreach(const CppcheckLibraryData::Function::Arg::MinSize &minsize, arg.minsizes) { QDomElement e = doc.createElement("minsize"); e.setAttribute("type", minsize.type); e.setAttribute("arg", minsize.arg); if (!minsize.arg2.isEmpty()) e.setAttribute("arg2", minsize.arg2); argElement.appendChild(e); } } } return functionElement; } static QDomElement MemoryResourceElement(QDomDocument &doc, const CppcheckLibraryData::MemoryResource &mr) { QDomElement element = doc.createElement(mr.type); foreach(const CppcheckLibraryData::MemoryResource::Alloc &alloc, mr.alloc) { QDomElement e = doc.createElement("alloc"); if (alloc.init) e.setAttribute("init", "true"); e.appendChild(doc.createTextNode(alloc.name)); element.appendChild(e); } foreach(const QString &dealloc, mr.dealloc) { QDomElement e = doc.createElement("dealloc"); e.appendChild(doc.createTextNode(dealloc)); element.appendChild(e); } foreach(const QString &use, mr.use) { QDomElement e = doc.createElement("use"); e.appendChild(doc.createTextNode(use)); element.appendChild(e); } return element; } QString CppcheckLibraryData::toString() const { QDomDocument doc; QDomElement root = doc.createElement("def"); doc.appendChild(root); root.setAttribute("format","2"); foreach(const Define &define, defines) { QDomElement defineElement = doc.createElement("define"); defineElement.setAttribute("name", define.name); defineElement.setAttribute("value", define.value); root.appendChild(defineElement); } foreach(const Function &function, functions) { foreach(const QString &comment, function.comments) { root.appendChild(doc.createComment(comment)); } root.appendChild(FunctionElement(doc, function)); } foreach(const MemoryResource &mr, memoryresource) { root.appendChild(MemoryResourceElement(doc, mr)); } foreach(const PodType &podtype, podtypes) { QDomElement podtypeElement = doc.createElement("podtype"); podtypeElement.setAttribute("name", podtype.name); if (!podtype.size.isEmpty()) podtypeElement.setAttribute("size", podtype.size); if (!podtype.sign.isEmpty()) podtypeElement.setAttribute("sign", podtype.sign); root.appendChild(podtypeElement); } return doc.toString(); }