12b8ee6c2994f738e5162ff46b638974870f51662Steve Naroff/* c-index-test.c */
250398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
3c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines#include "clang/Config/config.h"
450398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff#include "clang-c/Index.h"
5db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison#include "clang-c/CXCompilationDatabase.h"
6651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines#include "clang-c/BuildSystem.h"
76bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines#include "clang-c/Documentation.h"
81e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor#include <ctype.h>
90c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor#include <stdlib.h>
1089922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff#include <stdio.h>
11af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff#include <string.h>
12f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor#include <assert.h>
13af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff
14f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
15f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include <libxml/parser.h>
16f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include <libxml/relaxng.h>
17f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include <libxml/xmlerror.h>
18f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
19f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
20d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#ifdef _WIN32
21d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#  include <direct.h>
22d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#else
23d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#  include <unistd.h>
24d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#endif
25d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
260d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
270d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Utility functions.                                                         */
280d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
290d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
302e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#ifdef _MSC_VER
312e06fc877a633abea3b40a64950c7316dac29ca8John Thompsonchar *basename(const char* path)
322e06fc877a633abea3b40a64950c7316dac29ca8John Thompson{
332e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    char* base1 = (char*)strrchr(path, '/');
342e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    char* base2 = (char*)strrchr(path, '\\');
352e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    if (base1 && base2)
362e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return((base1 > base2) ? base1 + 1 : base2 + 1);
372e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    else if (base1)
382e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return(base1 + 1);
392e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    else if (base2)
402e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return(base2 + 1);
412e06fc877a633abea3b40a64950c7316dac29ca8John Thompson
422e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    return((char*)path);
432e06fc877a633abea3b40a64950c7316dac29ca8John Thompson}
44db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonchar *dirname(char* path)
45db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison{
46db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    char* base1 = (char*)strrchr(path, '/');
47db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    char* base2 = (char*)strrchr(path, '\\');
48db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    if (base1 && base2)
49db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (base1 > base2)
50db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          *base1 = 0;
51db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        else
52db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          *base2 = 0;
53db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    else if (base1)
540fb474a9323525964d437fefa22c06172492d270NAKAMURA Takumi        *base1 = 0;
55db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    else if (base2)
560fb474a9323525964d437fefa22c06172492d270NAKAMURA Takumi        *base2 = 0;
57db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
58db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    return path;
59db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison}
602e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#else
61ff9e18cb3863defa98257500ef49d93a9d2bc216Steve Naroffextern char *basename(const char *);
62db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonextern char *dirname(char *);
632e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#endif
64ff9e18cb3863defa98257500ef49d93a9d2bc216Steve Naroff
6545ba9a1b31110568d0d362c8d31c6133cf9011b7Douglas Gregor/** \brief Return the default parsing options. */
6644c181aec37789f25f6c15543c164416f72e562aDouglas Gregorstatic unsigned getDefaultParsingOptions() {
6744c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  unsigned options = CXTranslationUnit_DetailedPreprocessingRecord;
6844c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
6944c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  if (getenv("CINDEXTEST_EDITING"))
70b1c031be513705d924038f497279b9b599868ba1Douglas Gregor    options |= clang_defaultEditingTranslationUnitOptions();
7187c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor  if (getenv("CINDEXTEST_COMPLETION_CACHING"))
7287c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor    options |= CXTranslationUnit_CacheCompletionResults;
73dcaca015a2f08e03a6d86222fec2e83e04f6b848Argyrios Kyrtzidis  if (getenv("CINDEXTEST_COMPLETION_NO_CACHING"))
74dcaca015a2f08e03a6d86222fec2e83e04f6b848Argyrios Kyrtzidis    options &= ~CXTranslationUnit_CacheCompletionResults;
756a91d385618ea4d28236c496f540a26877c95525Erik Verbruggen  if (getenv("CINDEXTEST_SKIP_FUNCTION_BODIES"))
766a91d385618ea4d28236c496f540a26877c95525Erik Verbruggen    options |= CXTranslationUnit_SkipFunctionBodies;
77d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (getenv("CINDEXTEST_COMPLETION_BRIEF_COMMENTS"))
78d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    options |= CXTranslationUnit_IncludeBriefCommentsInCodeCompletion;
7944c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
8044c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  return options;
8144c181aec37789f25f6c15543c164416f72e562aDouglas Gregor}
8244c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
83651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines/** \brief Returns 0 in case of success, non-zero in case of a failure. */
84bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidisstatic int checkForErrors(CXTranslationUnit TU);
85bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
86651f13cea278ec967336033dd032faef0e9fc2ecStephen Hinesstatic void describeLibclangFailure(enum CXErrorCode Err) {
87651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  switch (Err) {
88651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  case CXError_Success:
89651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    fprintf(stderr, "Success\n");
90651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return;
91651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
92651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  case CXError_Failure:
93651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    fprintf(stderr, "Failure (no details available)\n");
94651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return;
95651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
96651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  case CXError_Crashed:
97651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    fprintf(stderr, "Failure: libclang crashed\n");
98651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return;
99651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
100651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  case CXError_InvalidArguments:
101651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    fprintf(stderr, "Failure: invalid arguments passed to a libclang routine\n");
102651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return;
103651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
104651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  case CXError_ASTReadError:
105651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    fprintf(stderr, "Failure: AST deserialization error occurred\n");
106651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return;
107651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  }
108651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines}
109651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
11051b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbarstatic void PrintExtent(FILE *out, unsigned begin_line, unsigned begin_column,
11151b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar                        unsigned end_line, unsigned end_column) {
11251b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  fprintf(out, "[%d:%d - %d:%d]", begin_line, begin_column,
113d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar          end_line, end_column);
11451b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar}
11551b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar
1161c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenekstatic unsigned CreateTranslationUnit(CXIndex Idx, const char *file,
1171c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek                                      CXTranslationUnit *TU) {
118651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err = clang_createTranslationUnit2(Idx, file, TU);
119651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
1201c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Unable to load translation unit from '%s'!\n", file);
121651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
122651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    *TU = 0;
1231c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 0;
124e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
1251c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  return 1;
1261c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
1271c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
1284db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregorvoid free_remapped_files(struct CXUnsavedFile *unsaved_files,
1294db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                         int num_unsaved_files) {
1304db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int i;
1314db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (i = 0; i != num_unsaved_files; ++i) {
1324db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    free((char *)unsaved_files[i].Filename);
1334db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    free((char *)unsaved_files[i].Contents);
1344db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
135653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor  free(unsaved_files);
1364db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor}
1374db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
138651f13cea278ec967336033dd032faef0e9fc2ecStephen Hinesstatic int parse_remapped_files_with_opt(const char *opt_name,
139651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         int argc, const char **argv,
140651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         int start_arg,
141651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         struct CXUnsavedFile **unsaved_files,
142651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         int *num_unsaved_files) {
1434db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int i;
1444db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int arg;
145651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int prefix_len = strlen(opt_name);
146651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int arg_indices[20];
1474db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *unsaved_files = 0;
1484db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *num_unsaved_files = 0;
149e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1504db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  /* Count the number of remapped files. */
1514db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (arg = start_arg; arg < argc; ++arg) {
152651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (strncmp(argv[arg], opt_name, prefix_len))
153651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      continue;
154e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
155651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    assert(*num_unsaved_files < (int)(sizeof(arg_indices)/sizeof(int)));
156651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    arg_indices[*num_unsaved_files] = arg;
1574db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    ++*num_unsaved_files;
1584db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
159e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1604db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  if (*num_unsaved_files == 0)
1614db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    return 0;
162e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1634db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *unsaved_files
164653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor    = (struct CXUnsavedFile *)malloc(sizeof(struct CXUnsavedFile) *
165653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor                                     *num_unsaved_files);
166651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  for (i = 0; i != *num_unsaved_files; ++i) {
1674db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    struct CXUnsavedFile *unsaved = *unsaved_files + i;
168651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    const char *arg_string = argv[arg_indices[i]] + prefix_len;
1694db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    int filename_len;
1704db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    char *filename;
1714db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    char *contents;
1724db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    FILE *to_file;
173651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    const char *sep = strchr(arg_string, ',');
174651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (!sep) {
175e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr,
176651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines              "error: %sfrom:to argument is missing comma\n", opt_name);
1774db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
1784db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
1794db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
1804db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
1814db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
182e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1834db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Open the file that we're remapping to. */
184651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    to_file = fopen(sep + 1, "rb");
1854db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (!to_file) {
1864db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fprintf(stderr, "error: cannot open file %s that we are remapping to\n",
187651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines              sep + 1);
1884db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
1894db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
1904db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
1914db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
1924db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
193e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1944db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Determine the length of the file we're remapping to. */
1954db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fseek(to_file, 0, SEEK_END);
1964db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Length = ftell(to_file);
1974db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fseek(to_file, 0, SEEK_SET);
198e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1994db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Read the contents of the file we're remapping to. */
2004db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    contents = (char *)malloc(unsaved->Length + 1);
2014db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (fread(contents, 1, unsaved->Length, to_file) != unsaved->Length) {
2024db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fprintf(stderr, "error: unexpected %s reading 'to' file %s\n",
203651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines              (feof(to_file) ? "EOF" : "error"), sep + 1);
2044db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fclose(to_file);
2054db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
206e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith      free(contents);
2074db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
2084db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
2094db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
2104db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
2114db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    contents[unsaved->Length] = 0;
2124db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Contents = contents;
213e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2144db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Close the file. */
2154db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fclose(to_file);
216e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2174db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Copy the file name that we're remapping from. */
218651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    filename_len = sep - arg_string;
2194db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    filename = (char *)malloc(filename_len + 1);
2204db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    memcpy(filename, arg_string, filename_len);
2214db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    filename[filename_len] = 0;
2224db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Filename = filename;
2234db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
224e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2254db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  return 0;
2264db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor}
2274db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
228651f13cea278ec967336033dd032faef0e9fc2ecStephen Hinesstatic int parse_remapped_files(int argc, const char **argv, int start_arg,
229651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                struct CXUnsavedFile **unsaved_files,
230651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                int *num_unsaved_files) {
231651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  return parse_remapped_files_with_opt("-remap-file=", argc, argv, start_arg,
232651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      unsaved_files, num_unsaved_files);
233651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines}
234651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
235651f13cea278ec967336033dd032faef0e9fc2ecStephen Hinesstatic int parse_remapped_files_with_try(int try_idx,
236651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         int argc, const char **argv,
237651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         int start_arg,
238651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         struct CXUnsavedFile **unsaved_files,
239651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         int *num_unsaved_files) {
240651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  struct CXUnsavedFile *unsaved_files_no_try_idx;
241651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int num_unsaved_files_no_try_idx;
242651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  struct CXUnsavedFile *unsaved_files_try_idx;
243651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int num_unsaved_files_try_idx;
244651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int ret;
245651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  char opt_name[32];
246651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
247651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  ret = parse_remapped_files(argc, argv, start_arg,
248651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      &unsaved_files_no_try_idx, &num_unsaved_files_no_try_idx);
249651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (ret)
250651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return ret;
251651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
252651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  sprintf(opt_name, "-remap-file-%d=", try_idx);
253651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  ret = parse_remapped_files_with_opt(opt_name, argc, argv, start_arg,
254651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      &unsaved_files_try_idx, &num_unsaved_files_try_idx);
255651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (ret)
256651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return ret;
257651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
258651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  *num_unsaved_files = num_unsaved_files_no_try_idx + num_unsaved_files_try_idx;
259651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  *unsaved_files
260651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    = (struct CXUnsavedFile *)realloc(unsaved_files_no_try_idx,
261651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                      sizeof(struct CXUnsavedFile) *
262651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                        *num_unsaved_files);
263651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  memcpy(*unsaved_files + num_unsaved_files_no_try_idx,
264651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines         unsaved_files_try_idx, sizeof(struct CXUnsavedFile) *
265651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines            num_unsaved_files_try_idx);
266651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  free(unsaved_files_try_idx);
267651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  return 0;
268651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines}
269651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
270f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkostatic const char *parse_comments_schema(int argc, const char **argv) {
271f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentsSchemaArg = "-comments-xml-schema=";
272f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentSchemaFile = NULL;
273f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
274f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (argc == 0)
275f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return CommentSchemaFile;
276f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
277f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!strncmp(argv[0], CommentsSchemaArg, strlen(CommentsSchemaArg)))
278f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    CommentSchemaFile = argv[0] + strlen(CommentsSchemaArg);
279f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
280f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  return CommentSchemaFile;
281f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko}
282f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
2830d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
2840d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Pretty-printing.                                                           */
2850d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
2860d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
287ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic const char *FileCheckPrefix = "CHECK";
288ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
289ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCString(const char *CStr) {
2902d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  if (CStr != NULL && CStr[0] != '\0') {
2912d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko    for ( ; *CStr; ++CStr) {
2922d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      const char C = *CStr;
2932d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      switch (C) {
2942d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\n': printf("\\n"); break;
2952d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\r': printf("\\r"); break;
2962d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\t': printf("\\t"); break;
2972d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\v': printf("\\v"); break;
2982d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\f': printf("\\f"); break;
2992d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        default:   putchar(C);    break;
3002d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      }
3012d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko    }
3022d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  }
303ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
304ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
305ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCStringWithPrefix(const char *Prefix, const char *CStr) {
306ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf(" %s=[", Prefix);
307ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCString(CStr);
3082d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  printf("]");
3092d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko}
3102d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko
311ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCXStringAndDispose(CXString Str) {
312ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCString(clang_getCString(Str));
313ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  clang_disposeString(Str);
314ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
315ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
316f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkostatic void PrintCXStringWithPrefix(const char *Prefix, CXString Str) {
317f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCStringWithPrefix(Prefix, clang_getCString(Str));
318f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko}
319f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
320ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCXStringWithPrefixAndDispose(const char *Prefix,
321ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                              CXString Str) {
322ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCStringWithPrefix(Prefix, clang_getCString(Str));
323ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  clang_disposeString(Str);
324ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
325ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
326430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregorstatic void PrintRange(CXSourceRange R, const char *str) {
327430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  CXFile begin_file, end_file;
328430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  unsigned begin_line, begin_column, end_line, end_column;
329430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
330430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  clang_getSpellingLocation(clang_getRangeStart(R),
331430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                            &begin_file, &begin_line, &begin_column, 0);
332430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  clang_getSpellingLocation(clang_getRangeEnd(R),
333430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                            &end_file, &end_line, &end_column, 0);
334430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  if (!begin_file || !end_file)
335430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    return;
336430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
337ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis  if (str)
338ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis    printf(" %s=", str);
339430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  PrintExtent(stdout, begin_line, begin_column, end_line, end_column);
340430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor}
341430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
342358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregorint want_display_name = 0;
343358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
344cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregorstatic void printVersion(const char *Prefix, CXVersion Version) {
345cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Major < 0)
346cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
347cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf("%s%d", Prefix, Version.Major);
348cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
349cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Minor < 0)
350cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
351cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf(".%d", Version.Minor);
352cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
353cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Subminor < 0)
354cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
355cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf(".%d", Version.Subminor);
356cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor}
357cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
358ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostruct CommentASTDumpingContext {
359ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  int IndentLevel;
360ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko};
361ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
362ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void DumpCXCommentInternal(struct CommentASTDumpingContext *Ctx,
363ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                  CXComment Comment) {
3645ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  unsigned i;
3655ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  unsigned e;
3665ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  enum CXCommentKind Kind = clang_Comment_getKind(Comment);
3675ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko
368ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx->IndentLevel++;
3695ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  for (i = 0, e = Ctx->IndentLevel; i != e; ++i)
370ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("  ");
371ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
372ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("(");
373ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  switch (Kind) {
374ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Null:
375ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Null");
376ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
377ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Text:
378ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Text");
379ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose("Text",
380ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                      clang_TextComment_getText(Comment));
381ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_isWhitespace(Comment))
382ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" IsWhitespace");
383ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
384ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
385ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
386ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_InlineCommand:
387ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_InlineCommand");
388ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
389ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
390ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_InlineCommandComment_getCommandName(Comment));
3912d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    switch (clang_InlineCommandComment_getRenderKind(Comment)) {
3922d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Normal:
3932d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderNormal");
3942d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
3952d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Bold:
3962d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderBold");
3972d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
3982d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Monospaced:
3992d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderMonospaced");
4002d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
4012d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Emphasized:
4022d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderEmphasized");
4032d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
4042d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    }
4055ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0, e = clang_InlineCommandComment_getNumArgs(Comment);
406ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko         i != e; ++i) {
407ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Arg[%u]=", i);
408ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringAndDispose(
409ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko          clang_InlineCommandComment_getArgText(Comment, i));
410ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
411ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
412ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
413ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
4145ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  case CXComment_HTMLStartTag: {
4155ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    unsigned NumAttrs;
416ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_HTMLStartTag");
417ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
418ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Name",
419ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_HTMLTagComment_getTagName(Comment));
4205ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    NumAttrs = clang_HTMLStartTag_getNumAttrs(Comment);
421ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (NumAttrs != 0) {
422ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Attrs:");
4235ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko      for (i = 0; i != NumAttrs; ++i) {
424ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        printf(" ");
425ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCXStringAndDispose(clang_HTMLStartTag_getAttrName(Comment, i));
426ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        printf("=");
427ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCXStringAndDispose(clang_HTMLStartTag_getAttrValue(Comment, i));
428ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      }
429ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
430ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_HTMLStartTagComment_isSelfClosing(Comment))
431ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" SelfClosing");
432ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
433ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
434ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
4355ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  }
436ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_HTMLEndTag:
437ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_HTMLEndTag");
438ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
439ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Name",
440ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_HTMLTagComment_getTagName(Comment));
441ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
442ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
443ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
444ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Paragraph:
445ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Paragraph");
446ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_isWhitespace(Comment))
447ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" IsWhitespace");
448ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
449ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_BlockCommand:
450ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_BlockCommand");
451ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
452ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
453ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_BlockCommandComment_getCommandName(Comment));
4545ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0, e = clang_BlockCommandComment_getNumArgs(Comment);
455ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko         i != e; ++i) {
456ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Arg[%u]=", i);
457ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringAndDispose(
458ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko          clang_BlockCommandComment_getArgText(Comment, i));
459ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
460ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
461ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_ParamCommand:
462ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_ParamCommand");
463ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    switch (clang_ParamCommandComment_getDirection(Comment)) {
464ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_In:
465ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" in");
466ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
467ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_Out:
468ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" out");
469ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
470ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_InOut:
471ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" in,out");
472ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
473ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
474ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_ParamCommandComment_isDirectionExplicit(Comment))
475ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" explicitly");
476ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    else
477ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" implicitly");
478ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
479ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "ParamName",
480ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_ParamCommandComment_getParamName(Comment));
481ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_ParamCommandComment_isParamIndexValid(Comment))
482ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" ParamIndex=%u", clang_ParamCommandComment_getParamIndex(Comment));
483ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    else
484ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" ParamIndex=Invalid");
485ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
48696b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko  case CXComment_TParamCommand:
48796b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    printf("CXComment_TParamCommand");
48896b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    PrintCXStringWithPrefixAndDispose(
48996b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        "ParamName",
49096b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        clang_TParamCommandComment_getParamName(Comment));
49196b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    if (clang_TParamCommandComment_isParamPositionValid(Comment)) {
49296b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf(" ParamPosition={");
49396b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      for (i = 0, e = clang_TParamCommandComment_getDepth(Comment);
49496b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko           i != e; ++i) {
49596b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        printf("%u", clang_TParamCommandComment_getIndex(Comment, i));
49696b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        if (i != e - 1)
49796b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko          printf(", ");
49896b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      }
49996b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf("}");
50096b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    } else
50196b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf(" ParamPosition=Invalid");
50296b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    break;
503ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimBlockCommand:
504ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimBlockCommand");
505ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
506ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
507ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_BlockCommandComment_getCommandName(Comment));
508ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
509ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimBlockLine:
510ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimBlockLine");
511ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
512ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Text",
513ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_VerbatimBlockLineComment_getText(Comment));
514ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
515ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimLine:
516ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimLine");
517ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
518ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Text",
519ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_VerbatimLineComment_getText(Comment));
520ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
521ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_FullComment:
522ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_FullComment");
523ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
524ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
525ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  if (Kind != CXComment_Null) {
526ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const unsigned NumChildren = clang_Comment_getNumChildren(Comment);
5275ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    unsigned i;
5285ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0; i != NumChildren; ++i) {
529ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf("\n// %s: ", FileCheckPrefix);
530ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      DumpCXCommentInternal(Ctx, clang_Comment_getChild(Comment, i));
531ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
532ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
533ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf(")");
534ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx->IndentLevel--;
535ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
536ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
537ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void DumpCXComment(CXComment Comment) {
538ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  struct CommentASTDumpingContext Ctx;
539ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx.IndentLevel = 1;
540ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("\n// %s:  CommentAST=[\n// %s:", FileCheckPrefix, FileCheckPrefix);
541ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  DumpCXCommentInternal(&Ctx, Comment);
542ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("]");
543ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
544ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
5456bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic void ValidateCommentXML(const char *Str, const char *CommentSchemaFile) {
546f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
547f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGParserCtxtPtr RNGParser;
548f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGPtr Schema;
549f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlDocPtr Doc;
550f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGValidCtxtPtr ValidationCtxt;
551f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  int status;
552f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
5536bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  if (!CommentSchemaFile)
554f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return;
555f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
5566bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  RNGParser = xmlRelaxNGNewParserCtxt(CommentSchemaFile);
5576bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  if (!RNGParser) {
558f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" libXMLError");
559f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return;
560f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
5616bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  Schema = xmlRelaxNGParse(RNGParser);
562f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
563f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  Doc = xmlParseDoc((const xmlChar *) Str);
564f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
565f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!Doc) {
566f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    xmlErrorPtr Error = xmlGetLastError();
567f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" CommentXMLInvalid [not well-formed XML: %s]", Error->message);
568f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return;
569f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
570f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
5716bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  ValidationCtxt = xmlRelaxNGNewValidCtxt(Schema);
572f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  status = xmlRelaxNGValidateDoc(ValidationCtxt, Doc);
573f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!status)
574f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" CommentXMLValid");
575f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  else if (status > 0) {
576f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    xmlErrorPtr Error = xmlGetLastError();
577f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" CommentXMLInvalid [not vaild XML: %s]", Error->message);
578f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  } else
579f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" libXMLError");
580f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
581f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGFreeValidCtxt(ValidationCtxt);
582f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlFreeDoc(Doc);
5836bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  xmlRelaxNGFree(Schema);
5846bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  xmlRelaxNGFreeParserCtxt(RNGParser);
585f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
586f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko}
587f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
588e4330a302ac20b41b9800267ebd4b5b01f8553f8Dmitri Gribenkostatic void PrintCursorComments(CXCursor Cursor,
5896bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                const char *CommentSchemaFile) {
590ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  {
591ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    CXString RawComment;
592ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const char *RawCommentCString;
593ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    CXString BriefComment;
594ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const char *BriefCommentCString;
595ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
596ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    RawComment = clang_Cursor_getRawCommentText(Cursor);
597ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    RawCommentCString = clang_getCString(RawComment);
598ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (RawCommentCString != NULL && RawCommentCString[0] != '\0') {
599ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCStringWithPrefix("RawComment", RawCommentCString);
600ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintRange(clang_Cursor_getCommentRange(Cursor), "RawCommentRange");
601ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
602ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      BriefComment = clang_Cursor_getBriefCommentText(Cursor);
603ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      BriefCommentCString = clang_getCString(BriefComment);
604ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      if (BriefCommentCString != NULL && BriefCommentCString[0] != '\0')
605ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCStringWithPrefix("BriefComment", BriefCommentCString);
606ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      clang_disposeString(BriefComment);
607ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
608ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    clang_disposeString(RawComment);
609ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
610ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
611ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  {
61210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXComment Comment = clang_Cursor_getParsedComment(Cursor);
613ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_getKind(Comment) != CXComment_Null) {
614ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringWithPrefixAndDispose("FullCommentAsHTML",
615ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                        clang_FullComment_getAsHTML(Comment));
616f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko      {
617f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        CXString XML;
618e4330a302ac20b41b9800267ebd4b5b01f8553f8Dmitri Gribenko        XML = clang_FullComment_getAsXML(Comment);
619f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCXStringWithPrefix("FullCommentAsXML", XML);
6206bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        ValidateCommentXML(clang_getCString(XML), CommentSchemaFile);
621f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        clang_disposeString(XML);
622f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko      }
623f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
624ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      DumpCXComment(Comment);
625ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
626ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
627ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
628ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
629b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidistypedef struct {
630b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  unsigned line;
631b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  unsigned col;
632b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis} LineCol;
633b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis
634b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidisstatic int lineCol_cmp(const void *p1, const void *p2) {
635b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  const LineCol *lhs = p1;
636b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  const LineCol *rhs = p2;
637b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  if (lhs->line != rhs->line)
638b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis    return (int)lhs->line - (int)rhs->line;
639b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  return (int)lhs->col - (int)rhs->col;
640b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis}
641b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis
6426bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic void PrintCursor(CXCursor Cursor, const char *CommentSchemaFile) {
643aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXTranslationUnit TU = clang_Cursor_getTranslationUnit(Cursor);
644e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (clang_isInvalid(Cursor.kind)) {
64510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString ks = clang_getCursorKindSpelling(Cursor.kind);
646e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("Invalid Cursor => %s", clang_getCString(ks));
647e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    clang_disposeString(ks);
648e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
649699a07d8a0b1579c5178b3baf4bcf9edb6b38108Steve Naroff  else {
650e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    CXString string, ks;
651c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    CXCursor Referenced;
6521db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    unsigned line, column;
653e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    CXCursor SpecializationOf;
6549f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    CXCursor *overridden;
6559f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    unsigned num_overridden;
656430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    unsigned RefNameRangeNr;
657430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CXSourceRange CursorExtent;
658430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CXSourceRange RefNameRange;
659cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int AlwaysUnavailable;
660cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int AlwaysDeprecated;
661cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXString UnavailableMessage;
662cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXString DeprecatedMessage;
663cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXPlatformAvailability PlatformAvailability[2];
664cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int NumPlatformAvailability;
665cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int I;
666aa0cd85838f2a024e589ea4e8c2094130065af21Dmitri Gribenko
667e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    ks = clang_getCursorKindSpelling(Cursor.kind);
668358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    string = want_display_name? clang_getCursorDisplayName(Cursor)
669358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                              : clang_getCursorSpelling(Cursor);
670e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("%s=%s", clang_getCString(ks),
671e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                    clang_getCString(string));
672e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    clang_disposeString(ks);
673ef0cef6cec8de5fc60e469a93436eed7212e0dc2Steve Naroff    clang_disposeString(string);
674e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
675c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    Referenced = clang_getCursorReferenced(Cursor);
676c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    if (!clang_equalCursors(Referenced, clang_getNullCursor())) {
6771f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (clang_getCursorKind(Referenced) == CXCursor_OverloadedDeclRef) {
6781f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        unsigned I, N = clang_getNumOverloadedDecls(Referenced);
6791f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf("[");
6801f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        for (I = 0; I != N; ++I) {
68110f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella          CXCursor Ovl = clang_getOverloadedDecl(Referenced, I);
6821f6206ed21e026ca7ddaf0bff9599c476301a695Douglas Gregor          CXSourceLocation Loc;
6831f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          if (I)
6841f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor            printf(", ");
6851f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
6861f6206ed21e026ca7ddaf0bff9599c476301a695Douglas Gregor          Loc = clang_getCursorLocation(Ovl);
687a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor          clang_getSpellingLocation(Loc, 0, &line, &column, 0);
6881f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          printf("%d:%d", line, column);
6891f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        }
6901f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf("]");
6911f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      } else {
69210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        CXSourceLocation Loc = clang_getCursorLocation(Referenced);
693a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor        clang_getSpellingLocation(Loc, 0, &line, &column, 0);
6941f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf(":%d:%d", line, column);
6951f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      }
696c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    }
697b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
698b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (clang_isCursorDefinition(Cursor))
699b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      printf(" (Definition)");
70058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
70158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    switch (clang_getCursorAvailability(Cursor)) {
70258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_Available:
70358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
70458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
70558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_Deprecated:
70658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        printf(" (deprecated)");
70758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
70858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
70958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_NotAvailable:
71058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        printf(" (unavailable)");
71158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
712d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen
713d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen      case CXAvailability_NotAccessible:
714d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen        printf(" (inaccessible)");
715d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen        break;
71658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    }
71795f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
718cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    NumPlatformAvailability
719cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      = clang_getCursorPlatformAvailability(Cursor,
720cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &AlwaysDeprecated,
721cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &DeprecatedMessage,
722cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &AlwaysUnavailable,
723cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &UnavailableMessage,
724cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            PlatformAvailability, 2);
725cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    if (AlwaysUnavailable) {
726cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      printf("  (always unavailable: \"%s\")",
727cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor             clang_getCString(UnavailableMessage));
728cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    } else if (AlwaysDeprecated) {
729cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      printf("  (always deprecated: \"%s\")",
730cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor             clang_getCString(DeprecatedMessage));
731cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    } else {
732cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      for (I = 0; I != NumPlatformAvailability; ++I) {
733cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (I >= 2)
734cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          break;
735cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
736cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        printf("  (%s", clang_getCString(PlatformAvailability[I].Platform));
737cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (PlatformAvailability[I].Unavailable)
738cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printf(", unavailable");
739cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        else {
740cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", introduced=", PlatformAvailability[I].Introduced);
741cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", deprecated=", PlatformAvailability[I].Deprecated);
742cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", obsoleted=", PlatformAvailability[I].Obsoleted);
743cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        }
744cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (clang_getCString(PlatformAvailability[I].Message)[0])
745cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printf(", message=\"%s\"",
746cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                 clang_getCString(PlatformAvailability[I].Message));
747cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        printf(")");
748cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      }
749cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    }
750cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    for (I = 0; I != NumPlatformAvailability; ++I) {
751cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      if (I >= 2)
752cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        break;
753cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      clang_disposeCXPlatformAvailability(PlatformAvailability + I);
754cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    }
755cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
756cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    clang_disposeString(DeprecatedMessage);
757cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    clang_disposeString(UnavailableMessage);
758cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
759b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor    if (clang_CXXMethod_isStatic(Cursor))
760b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor      printf(" (static)");
761b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor    if (clang_CXXMethod_isVirtual(Cursor))
762b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor      printf(" (virtual)");
7636bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    if (clang_CXXMethod_isConst(Cursor))
7646bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      printf(" (const)");
765c965f76e02d536d875b1e9e3ea96f39452c6bfa2Dmitri Gribenko    if (clang_CXXMethod_isPureVirtual(Cursor))
766c965f76e02d536d875b1e9e3ea96f39452c6bfa2Dmitri Gribenko      printf(" (pure)");
76780e1acaeb7040548cb494e609b120b134e22a193Argyrios Kyrtzidis    if (clang_Cursor_isVariadic(Cursor))
76880e1acaeb7040548cb494e609b120b134e22a193Argyrios Kyrtzidis      printf(" (variadic)");
769514afc7255a2aaa88498b3374c944a2c497c1c55Argyrios Kyrtzidis    if (clang_Cursor_isObjCOptional(Cursor))
770514afc7255a2aaa88498b3374c944a2c497c1c55Argyrios Kyrtzidis      printf(" (@optional)");
771514afc7255a2aaa88498b3374c944a2c497c1c55Argyrios Kyrtzidis
77295f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek    if (Cursor.kind == CXCursor_IBOutletCollectionAttr) {
77310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXType T =
77410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        clang_getCanonicalType(clang_getIBOutletCollectionType(Cursor));
77510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXString S = clang_getTypeKindSpelling(T.kind);
77695f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      printf(" [IBOutletCollection=%s]", clang_getCString(S));
77795f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      clang_disposeString(S);
77895f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek    }
7793064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
7803064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    if (Cursor.kind == CXCursor_CXXBaseSpecifier) {
7813064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      enum CX_CXXAccessSpecifier access = clang_getCXXAccessSpecifier(Cursor);
7823064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      unsigned isVirtual = clang_isVirtualBase(Cursor);
7833064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      const char *accessStr = 0;
7843064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
7853064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      switch (access) {
7863064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXInvalidAccessSpecifier:
7873064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "invalid"; break;
7883064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXPublic:
7893064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "public"; break;
7903064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXProtected:
7913064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "protected"; break;
7923064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXPrivate:
7933064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "private"; break;
7943064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      }
7953064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
7963064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      printf(" [access=%s isVirtual=%s]", accessStr,
7973064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek             isVirtual ? "true" : "false");
7983064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
799176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines
800e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    SpecializationOf = clang_getSpecializedCursorTemplate(Cursor);
801e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    if (!clang_equalCursors(SpecializationOf, clang_getNullCursor())) {
80210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXSourceLocation Loc = clang_getCursorLocation(SpecializationOf);
80310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXString Name = clang_getCursorSpelling(SpecializationOf);
804a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor      clang_getSpellingLocation(Loc, 0, &line, &column, 0);
805176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines      printf(" [Specialization of %s:%d:%d]",
806e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor             clang_getCString(Name), line, column);
807e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      clang_disposeString(Name);
808176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines
809176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines      if (Cursor.kind == CXCursor_FunctionDecl) {
810176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines        /* Collect the template parameter kinds from the base template. */
811176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines        unsigned NumTemplateArgs = clang_Cursor_getNumTemplateArguments(Cursor);
812176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines        unsigned I;
813176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines        for (I = 0; I < NumTemplateArgs; I++) {
814176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines          enum CXTemplateArgumentKind TAK =
815176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              clang_Cursor_getTemplateArgumentKind(Cursor, I);
816176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines          switch(TAK) {
817176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines            case CXTemplateArgumentKind_Type:
818176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              {
819176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                CXType T = clang_Cursor_getTemplateArgumentType(Cursor, I);
820176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                CXString S = clang_getTypeSpelling(T);
821176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                printf(" [Template arg %d: kind: %d, type: %s]",
822176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                       I, TAK, clang_getCString(S));
823176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                clang_disposeString(S);
824176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              }
825176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              break;
826176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines            case CXTemplateArgumentKind_Integral:
827176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              printf(" [Template arg %d: kind: %d, intval: %lld]",
828176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                     I, TAK, clang_Cursor_getTemplateArgumentValue(Cursor, I));
829176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              break;
830176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines            default:
831176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines              printf(" [Template arg %d: kind: %d]\n", I, TAK);
832176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines          }
833176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines        }
834176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines      }
835e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    }
8369f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
8379f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    clang_getOverriddenCursors(Cursor, &overridden, &num_overridden);
8389f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    if (num_overridden) {
8399f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      unsigned I;
840b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      LineCol lineCols[50];
841b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      assert(num_overridden <= 50);
8429f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      printf(" [Overrides ");
8439f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      for (I = 0; I != num_overridden; ++I) {
84410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        CXSourceLocation Loc = clang_getCursorLocation(overridden[I]);
845a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor        clang_getSpellingLocation(Loc, 0, &line, &column, 0);
846b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        lineCols[I].line = line;
847b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        lineCols[I].col = column;
848b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      }
849642214936430d2258318f2022184c1709dfa16d3Michael Liao      /* Make the order of the override list deterministic. */
850b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      qsort(lineCols, num_overridden, sizeof(LineCol), lineCol_cmp);
851b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      for (I = 0; I != num_overridden; ++I) {
8529f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor        if (I)
8539f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor          printf(", ");
854b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        printf("@%d:%d", lineCols[I].line, lineCols[I].col);
8559f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      }
8569f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      printf("]");
8579f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      clang_disposeOverriddenCursors(overridden);
8589f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    }
859ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
860ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    if (Cursor.kind == CXCursor_InclusionDirective) {
86110f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXFile File = clang_getIncludedFile(Cursor);
86210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXString Included = clang_getFileName(File);
863ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      printf(" (%s)", clang_getCString(Included));
864ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      clang_disposeString(Included);
865dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor
866dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor      if (clang_isFileMultipleIncludeGuarded(TU, File))
867dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor        printf("  [multi-include guarded]");
868ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    }
869430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
870430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CursorExtent = clang_getCursorExtent(Cursor);
871430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    RefNameRange = clang_getCursorReferenceNameRange(Cursor,
872430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                   CXNameRange_WantQualifier
873430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantSinglePiece
874430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantTemplateArgs,
875430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                     0);
876430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (!clang_equalRanges(CursorExtent, RefNameRange))
877430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      PrintRange(RefNameRange, "SingleRefName");
878430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
879430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    for (RefNameRangeNr = 0; 1; RefNameRangeNr++) {
880430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      RefNameRange = clang_getCursorReferenceNameRange(Cursor,
881430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                   CXNameRange_WantQualifier
882430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantTemplateArgs,
883430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                       RefNameRangeNr);
884430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (clang_equalRanges(clang_getNullRange(), RefNameRange))
885430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        break;
886430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (!clang_equalRanges(CursorExtent, RefNameRange))
887430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        PrintRange(RefNameRange, "RefName");
888430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    }
889aa0cd85838f2a024e589ea4e8c2094130065af21Dmitri Gribenko
8906bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    PrintCursorComments(Cursor, CommentSchemaFile);
8919ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis
8929ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis    {
8939ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis      unsigned PropAttrs = clang_Cursor_getObjCPropertyAttributes(Cursor, 0);
8949ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis      if (PropAttrs != CXObjCPropertyAttr_noattr) {
8959ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        printf(" [");
8969ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        #define PRINT_PROP_ATTR(A) \
8979ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis          if (PropAttrs & CXObjCPropertyAttr_##A) printf(#A ",")
8989ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(readonly);
8999ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(getter);
9009ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(assign);
9019ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(readwrite);
9029ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(retain);
9039ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(copy);
9049ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(nonatomic);
9059ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(setter);
9069ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(atomic);
9079ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(weak);
9089ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(strong);
9099ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(unsafe_unretained);
9109ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        printf("]");
9119ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis      }
9129ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis    }
91338dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis
91438dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis    {
91538dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis      unsigned QT = clang_Cursor_getObjCDeclQualifiers(Cursor);
91638dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis      if (QT != CXObjCDeclQualifier_None) {
91738dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        printf(" [");
91838dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        #define PRINT_OBJC_QUAL(A) \
91938dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis          if (QT & CXObjCDeclQualifier_##A) printf(#A ",")
92038dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(In);
92138dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Inout);
92238dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Out);
92338dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Bycopy);
92438dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Byref);
92538dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Oneway);
92638dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        printf("]");
92738dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis      }
92838dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis    }
929699a07d8a0b1579c5178b3baf4bcf9edb6b38108Steve Naroff  }
930af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff}
93189922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff
932e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekstatic const char* GetCursorSource(CXCursor Cursor) {
93310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXSourceLocation Loc = clang_getCursorLocation(Cursor);
93474844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  CXString source;
9351db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  CXFile file;
936b4efaa0a14dd2382aa028c03283b5a7f5345e24dArgyrios Kyrtzidis  clang_getExpansionLocation(Loc, &file, 0, 0, 0);
9371db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  source = clang_getFileName(file);
938e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (!clang_getCString(source)) {
93974844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
94074844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    return "<invalid loc>";
94174844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  }
94274844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  else {
943e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    const char *b = basename(clang_getCString(source));
94474844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
94574844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    return b;
94674844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  }
9479298cfc7475c48fa42b168c37f628663d65ddde7Ted Kremenek}
9489298cfc7475c48fa42b168c37f628663d65ddde7Ted Kremenek
9490d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
950ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/* Callbacks.                                                                 */
951ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/******************************************************************************/
952ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek
953ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenektypedef void (*PostVisitTU)(CXTranslationUnit);
954ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek
955a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregorvoid PrintDiagnostic(CXDiagnostic Diagnostic) {
956a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  FILE *out = stderr;
9575352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  CXFile file;
958274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  CXString Msg;
9590a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  unsigned display_opts = CXDiagnostic_DisplaySourceLocation
960aa5f135f8db82b5e5fb1640fd51f8078e0b2d82dDouglas Gregor    | CXDiagnostic_DisplayColumn | CXDiagnostic_DisplaySourceRanges
961aa5f135f8db82b5e5fb1640fd51f8078e0b2d82dDouglas Gregor    | CXDiagnostic_DisplayOption;
9620a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  unsigned i, num_fixits;
963f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
9640a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (clang_getDiagnosticSeverity(Diagnostic) == CXDiagnostic_Ignored)
9655352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return;
966e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
967274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  Msg = clang_formatDiagnostic(Diagnostic, display_opts);
968274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  fprintf(stderr, "%s\n", clang_getCString(Msg));
969274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  clang_disposeString(Msg);
970f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
971a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  clang_getSpellingLocation(clang_getDiagnosticLocation(Diagnostic),
972a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                            &file, 0, 0, 0);
9730a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (!file)
9740a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    return;
975e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
9760a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  num_fixits = clang_getDiagnosticNumFixIts(Diagnostic);
9773739b32a01b9778176757425e62252c288b85383Ted Kremenek  fprintf(stderr, "Number FIX-ITs = %d\n", num_fixits);
9780a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  for (i = 0; i != num_fixits; ++i) {
979473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXSourceRange range;
98010f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString insertion_text = clang_getDiagnosticFixIt(Diagnostic, i, &range);
98110f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceLocation start = clang_getRangeStart(range);
98210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceLocation end = clang_getRangeEnd(range);
983473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    unsigned start_line, start_column, end_line, end_column;
984473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXFile start_file, end_file;
985a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(start, &start_file, &start_line,
986a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              &start_column, 0);
987a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(end, &end_file, &end_line, &end_column, 0);
988473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    if (clang_equalLocations(start, end)) {
989473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Insertion. */
990473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      if (start_file == file)
9910a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Insert \"%s\" at %d:%d\n",
992473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor                clang_getCString(insertion_text), start_line, start_column);
993473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    } else if (strcmp(clang_getCString(insertion_text), "") == 0) {
994473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Removal. */
9950a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      if (start_file == file && end_file == file) {
9960a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Remove ");
9970a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        PrintExtent(out, start_line, start_column, end_line, end_column);
9980a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "\n");
999436f3f0400c633251e4071f81358c47bab964adfDouglas Gregor      }
1000473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    } else {
1001473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Replacement. */
10020a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      if (start_file == end_file) {
10030a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Replace ");
10040a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        PrintExtent(out, start_line, start_column, end_line, end_column);
1005473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor        fprintf(out, " with \"%s\"\n", clang_getCString(insertion_text));
100651c6d384551674facc19f745ecf6e289d28dc55fDouglas Gregor      }
10070a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    }
1008473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    clang_disposeString(insertion_text);
100951c6d384551674facc19f745ecf6e289d28dc55fDouglas Gregor  }
10105352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor}
10115352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor
10127473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenekvoid PrintDiagnosticSet(CXDiagnosticSet Set) {
10137473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  int i = 0, n = clang_getNumDiagnosticsInSet(Set);
10147473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  for ( ; i != n ; ++i) {
10157473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    CXDiagnostic Diag = clang_getDiagnosticInSet(Set, i);
10167473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    CXDiagnosticSet ChildDiags = clang_getChildDiagnostics(Diag);
1017a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    PrintDiagnostic(Diag);
10187473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    if (ChildDiags)
10197473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek      PrintDiagnosticSet(ChildDiags);
10207473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  }
10217473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek}
10227473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek
10237473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenekvoid PrintDiagnostics(CXTranslationUnit TU) {
10247473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  CXDiagnosticSet TUSet = clang_getDiagnosticSetFromTU(TU);
10257473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  PrintDiagnosticSet(TUSet);
10267473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  clang_disposeDiagnosticSet(TUSet);
1027a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor}
1028a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor
102959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenekvoid PrintMemoryUsage(CXTranslationUnit TU) {
1030b22732365ca26fbccacc4bdd32e875342f571e94Matt Beaumont-Gay  unsigned long total = 0;
10314e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  unsigned i = 0;
103210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXTUResourceUsage usage = clang_getCXTUResourceUsage(TU);
10333c683367073e2d98a9046060f9bc7db872a1c63dFrancois Pichet  fprintf(stderr, "Memory usage:\n");
10344e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  for (i = 0 ; i != usage.numEntries; ++i) {
1035f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    const char *name = clang_getTUResourceUsageName(usage.entries[i].kind);
103659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    unsigned long amount = usage.entries[i].amount;
103759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    total += amount;
10384e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    fprintf(stderr, "  %s : %ld bytes (%f MBytes)\n", name, amount,
103959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek            ((double) amount)/(1024*1024));
104059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  }
10414e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  fprintf(stderr, "  TOTAL = %ld bytes (%f MBytes)\n", total,
104259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek          ((double) total)/(1024*1024));
1043f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  clang_disposeCXTUResourceUsage(usage);
104459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
104559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
1046ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/******************************************************************************/
1047e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor/* Logic for testing traversal.                                               */
10480d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
10490d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
1050a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregorstatic void PrintCursorExtent(CXCursor C) {
105110f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXSourceRange extent = clang_getCursorExtent(C);
105210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  PrintRange(extent, "Extent");
1053fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek}
1054fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek
1055f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko/* Data used by the visitors. */
1056f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkotypedef struct {
1057e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  CXTranslationUnit TU;
1058e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  enum CXCursorKind *Filter;
10596bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  const char *CommentSchemaFile;
1060e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor} VisitorData;
1061fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek
1062625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1063e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekenum CXChildVisitResult FilteredPrintingVisitor(CXCursor Cursor,
1064e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                CXCursor Parent,
1065e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                CXClientData ClientData) {
1066e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
1067e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  if (!Data->Filter || (Cursor.kind == *(enum CXCursorKind *)Data->Filter)) {
106810f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceLocation Loc = clang_getCursorLocation(Cursor);
10691db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    unsigned line, column;
1070a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(Loc, 0, &line, &column, 0);
1071fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek    printf("// %s: %s:%d:%d: ", FileCheckPrefix,
10721db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor           GetCursorSource(Cursor), line, column);
10736bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    PrintCursor(Cursor, Data->CommentSchemaFile);
1074a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    PrintCursorExtent(Cursor);
107504b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis    if (clang_isDeclaration(Cursor.kind)) {
107604b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      enum CX_CXXAccessSpecifier access = clang_getCXXAccessSpecifier(Cursor);
107704b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      const char *accessStr = 0;
107804b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis
107904b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      switch (access) {
108004b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        case CX_CXXInvalidAccessSpecifier: break;
108104b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        case CX_CXXPublic:
108204b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis          accessStr = "public"; break;
108304b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        case CX_CXXProtected:
108404b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis          accessStr = "protected"; break;
108504b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        case CX_CXXPrivate:
108604b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis          accessStr = "private"; break;
108704b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      }
108804b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis
108904b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      if (accessStr)
109004b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        printf(" [access=%s]", accessStr);
109104b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis    }
1092e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("\n");
1093e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Recurse;
1094625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  }
1095e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1096e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
1097625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar}
1098c857ea4d22e1e6dd9ede1f0e84d48b157c6924fdSteve Naroff
1099e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekstatic enum CXChildVisitResult FunctionScanVisitor(CXCursor Cursor,
1100e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                   CXCursor Parent,
1101e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                   CXClientData ClientData) {
1102625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  const char *startBuf, *endBuf;
1103625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  unsigned startLine, startColumn, endLine, endColumn, curLine, curColumn;
1104625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXCursor Ref;
1105e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
1106625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1107b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (Cursor.kind != CXCursor_FunctionDecl ||
1108b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      !clang_isCursorDefinition(Cursor))
1109e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Continue;
1110625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1111625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  clang_getDefinitionSpellingAndExtent(Cursor, &startBuf, &endBuf,
1112625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar                                       &startLine, &startColumn,
1113625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar                                       &endLine, &endColumn);
1114625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  /* Probe the entire body, looking for both decls and refs. */
1115625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  curLine = startLine;
1116625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  curColumn = startColumn;
1117625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1118625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  while (startBuf < endBuf) {
111998258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    CXSourceLocation Loc;
11201db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    CXFile file;
112174844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    CXString source;
1122e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1123625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    if (*startBuf == '\n') {
1124625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      startBuf++;
1125625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curLine++;
1126625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curColumn = 1;
1127625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    } else if (*startBuf != '\t')
1128625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curColumn++;
1129e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
113098258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    Loc = clang_getCursorLocation(Cursor);
1131a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(Loc, &file, 0, 0, 0);
1132e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
11331db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    source = clang_getFileName(file);
1134e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    if (clang_getCString(source)) {
113510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXSourceLocation RefLoc
113610f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        = clang_getLocation(Data->TU, file, curLine, curColumn);
1137b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor      Ref = clang_getCursor(Data->TU, RefLoc);
113898258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      if (Ref.kind == CXCursor_NoDeclFound) {
113998258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        /* Nothing found here; that's fine. */
114098258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      } else if (Ref.kind != CXCursor_FunctionDecl) {
114198258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        printf("// %s: %s:%d:%d: ", FileCheckPrefix, GetCursorSource(Ref),
114298258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor               curLine, curColumn);
11436bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        PrintCursor(Ref, Data->CommentSchemaFile);
114498258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        printf("\n");
114598258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      }
11464ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff    }
114774844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
1148625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    startBuf++;
11492d4d629d8a0de5112c7ae9d05c03ddbf6dcd956aSteve Naroff  }
1150e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1151e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
115289922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff}
115350398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
11547d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
11557d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/* USR testing.                                                               */
11567d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
11577d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
1158e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregorenum CXChildVisitResult USRVisitor(CXCursor C, CXCursor parent,
1159e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                   CXClientData ClientData) {
1160e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
1161e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  if (!Data->Filter || (C.kind == *(enum CXCursorKind *)Data->Filter)) {
116210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString USR = clang_getCursorUSR(C);
116310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    const char *cstr = clang_getCString(USR);
1164e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    if (!cstr || cstr[0] == '\0') {
11657d40562f83552b7295411e10ee887d8d55470679Ted Kremenek      clang_disposeString(USR);
1166e74ef1289d5fff0a6ea573198bf354fa8cd84d51Ted Kremenek      return CXChildVisit_Recurse;
11677d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    }
1168e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    printf("// %s: %s %s", FileCheckPrefix, GetCursorSource(C), cstr);
1169e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek
1170a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    PrintCursorExtent(C);
11717d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    printf("\n");
11727d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    clang_disposeString(USR);
1173e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1174e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Recurse;
1175e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
1176e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1177e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
11787d40562f83552b7295411e10ee887d8d55470679Ted Kremenek}
11797d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
11807d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
118116b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/* Inclusion stack testing.                                                   */
118216b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/******************************************************************************/
118316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
118416b55a71695a33c094383295cc7b7a2080e098daTed Kremenekvoid InclusionVisitor(CXFile includedFile, CXSourceLocation *includeStack,
118516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                      unsigned includeStackLen, CXClientData data) {
1186e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
118716b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  unsigned i;
118874844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  CXString fname;
118974844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek
119074844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  fname = clang_getFileName(includedFile);
1191e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  printf("file: %s\nincluded by:\n", clang_getCString(fname));
119274844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  clang_disposeString(fname);
1193e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
119416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  for (i = 0; i < includeStackLen; ++i) {
119516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    CXFile includingFile;
119616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    unsigned line, column;
1197a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(includeStack[i], &includingFile, &line,
1198a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              &column, 0);
119974844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    fname = clang_getFileName(includingFile);
1200e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("  %s:%d:%d\n", clang_getCString(fname), line, column);
120174844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(fname);
120216b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  }
120316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  printf("\n");
120416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek}
120516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
120616b55a71695a33c094383295cc7b7a2080e098daTed Kremenekvoid PrintInclusionStack(CXTranslationUnit TU) {
1207e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  clang_getInclusions(TU, InclusionVisitor, NULL);
120816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek}
120916b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
121016b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/******************************************************************************/
12113bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/* Linkage testing.                                                           */
12123bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/******************************************************************************/
12133bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12143bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenekstatic enum CXChildVisitResult PrintLinkage(CXCursor cursor, CXCursor p,
12153bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek                                            CXClientData d) {
12163bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  const char *linkage = 0;
12173bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12183bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  if (clang_isInvalid(clang_getCursorKind(cursor)))
12193bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    return CXChildVisit_Recurse;
12203bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12213bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  switch (clang_getCursorLinkage(cursor)) {
12223bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    case CXLinkage_Invalid: break;
1223c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_NoLinkage: linkage = "NoLinkage"; break;
1224c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_Internal: linkage = "Internal"; break;
1225c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_UniqueExternal: linkage = "UniqueExternal"; break;
1226c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_External: linkage = "External"; break;
12273bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  }
12283bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12293bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  if (linkage) {
1230f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(cursor, NULL);
12313bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    printf("linkage=%s\n", linkage);
12323bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  }
12333bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12343bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  return CXChildVisit_Recurse;
12353bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek}
12363bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12373bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/******************************************************************************/
12388e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/* Typekind testing.                                                          */
12398e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/******************************************************************************/
12408e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
1241ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenkostatic void PrintTypeAndTypeKind(CXType T, const char *Format) {
1242ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  CXString TypeSpelling, TypeKindSpelling;
1243ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko
1244ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  TypeSpelling = clang_getTypeSpelling(T);
1245ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  TypeKindSpelling = clang_getTypeKindSpelling(T.kind);
1246ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  printf(Format,
1247ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko         clang_getCString(TypeSpelling),
1248ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko         clang_getCString(TypeKindSpelling));
1249ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  clang_disposeString(TypeSpelling);
1250ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  clang_disposeString(TypeKindSpelling);
1251ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko}
1252ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko
125333337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainarstatic enum CXVisitorResult FieldVisitor(CXCursor C,
125433337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar                                         CXClientData client_data) {
125533337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar    (*(int *) client_data)+=1;
125633337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar    return CXVisit_Continue;
125733337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar}
125833337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar
1259ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenkostatic enum CXChildVisitResult PrintType(CXCursor cursor, CXCursor p,
1260ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko                                         CXClientData d) {
12618e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  if (!clang_isInvalid(clang_getCursorKind(cursor))) {
126210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXType T = clang_getCursorType(cursor);
1263659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis    enum CXRefQualifierKind RQ = clang_Type_getCXXRefQualifier(T);
1264f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(cursor, NULL);
1265ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko    PrintTypeAndTypeKind(T, " [type=%s] [typekind=%s]");
1266e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isConstQualifiedType(T))
1267e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" const");
1268e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isVolatileQualifiedType(T))
1269e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" volatile");
1270e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isRestrictQualifiedType(T))
1271e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" restrict");
1272659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis    if (RQ == CXRefQualifier_LValue)
1273659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis      printf(" lvalue-ref-qualifier");
1274659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis    if (RQ == CXRefQualifier_RValue)
1275659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis      printf(" rvalue-ref-qualifier");
1276e1403d2dcb06fc306e186c787a7bd0ca30e06edeBenjamin Kramer    /* Print the canonical type if it is different. */
127704c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    {
127810f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXType CT = clang_getCanonicalType(T);
127904c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      if (!clang_equalTypes(T, CT)) {
1280ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko        PrintTypeAndTypeKind(CT, " [canonicaltype=%s] [canonicaltypekind=%s]");
128104c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      }
128204c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    }
1283e1403d2dcb06fc306e186c787a7bd0ca30e06edeBenjamin Kramer    /* Print the return type if it exists. */
128404c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    {
128510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXType RT = clang_getCursorResultType(cursor);
128604c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      if (RT.kind != CXType_Invalid) {
1287ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko        PrintTypeAndTypeKind(RT, " [resulttype=%s] [resulttypekind=%s]");
128804c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      }
128904c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    }
1290d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    /* Print the argument types if they exist. */
1291d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    {
1292651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      int NumArgs = clang_Cursor_getNumArguments(cursor);
1293651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (NumArgs != -1 && NumArgs != 0) {
129447f1165c92bc4104e314223ed9ad251e914687c1Argyrios Kyrtzidis        int i;
1295d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        printf(" [args=");
1296651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        for (i = 0; i < NumArgs; ++i) {
129710f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella          CXType T = clang_getCursorType(clang_Cursor_getArgument(cursor, i));
1298d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          if (T.kind != CXType_Invalid) {
1299ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko            PrintTypeAndTypeKind(T, " [%s] [%s]");
1300d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          }
1301d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        }
1302d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        printf("]");
1303d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis      }
1304d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    }
1305651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    /* Print the template argument types if they exist. */
1306651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    {
1307651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      int NumTArgs = clang_Type_getNumTemplateArguments(T);
1308651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (NumTArgs != -1 && NumTArgs != 0) {
1309651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        int i;
1310651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        printf(" [templateargs/%d=", NumTArgs);
1311651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        for (i = 0; i < NumTArgs; ++i) {
1312651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          CXType TArg = clang_Type_getTemplateArgumentAsType(T, i);
1313651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          if (TArg.kind != CXType_Invalid) {
1314651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines            PrintTypeAndTypeKind(TArg, " [type=%s] [typekind=%s]");
1315651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          }
1316651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        }
1317651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        printf("]");
1318651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      }
1319651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    }
13203ce9e7d270e7df86c09c8126b4412d55be7c123bTed Kremenek    /* Print if this is a non-POD type. */
13213ce9e7d270e7df86c09c8126b4412d55be7c123bTed Kremenek    printf(" [isPOD=%d]", clang_isPODType(T));
13226bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    /* Print the pointee type. */
13236bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    {
13246bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      CXType PT = clang_getPointeeType(T);
13256bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      if (PT.kind != CXType_Invalid) {
13266bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        PrintTypeAndTypeKind(PT, " [pointeetype=%s] [pointeekind=%s]");
13276bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      }
13286bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    }
132933337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar    /* Print the number of fields if they exist. */
133033337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar    {
133133337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar      int numFields = 0;
133233337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar      if (clang_Type_visitFields(T, FieldVisitor, &numFields)){
133333337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        if (numFields != 0) {
133433337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar          printf(" [nbFields=%d]", numFields);
133533337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        }
133633337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        /* Print if it is an anonymous record. */
133733337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        {
133833337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar          unsigned isAnon = clang_Cursor_isAnonymous(cursor);
133933337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar          if (isAnon != 0) {
134033337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar            printf(" [isAnon=%d]", isAnon);
134133337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar          }
134233337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        }
134333337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar      }
134433337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar    }
134504c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek
13468e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    printf("\n");
13478e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  }
13488e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  return CXChildVisit_Recurse;
13498e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek}
13508e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
1351411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidisstatic enum CXChildVisitResult PrintTypeSize(CXCursor cursor, CXCursor p,
1352411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis                                             CXClientData d) {
1353411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  CXType T;
1354411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  enum CXCursorKind K = clang_getCursorKind(cursor);
1355411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  if (clang_isInvalid(K))
1356411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    return CXChildVisit_Recurse;
1357411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  T = clang_getCursorType(cursor);
1358411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  PrintCursor(cursor, NULL);
1359411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  PrintTypeAndTypeKind(T, " [type=%s] [typekind=%s]");
1360411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  /* Print the type sizeof if applicable. */
1361411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  {
1362411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    long long Size = clang_Type_getSizeOf(T);
1363411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    if (Size >= 0 || Size < -1 ) {
1364411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      printf(" [sizeof=%lld]", Size);
1365411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    }
1366411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  }
1367411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  /* Print the type alignof if applicable. */
1368411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  {
1369411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    long long Align = clang_Type_getAlignOf(T);
1370411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    if (Align >= 0 || Align < -1) {
1371411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      printf(" [alignof=%lld]", Align);
1372411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    }
1373411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  }
1374411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  /* Print the record field offset if applicable. */
1375411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  {
13766bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    CXString FieldSpelling = clang_getCursorSpelling(cursor);
13776bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    const char *FieldName = clang_getCString(FieldSpelling);
137833337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar    /* recurse to get the first parent record that is not anonymous. */
137933337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar    CXCursor Parent, Record;
138033337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar    unsigned RecordIsAnonymous = 0;
13816bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    if (clang_getCursorKind(cursor) == CXCursor_FieldDecl) {
138233337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar      Record = Parent = p;
1383411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      do {
138433337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        Record = Parent;
138533337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        Parent = clang_getCursorSemanticParent(Record);
138633337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        RecordIsAnonymous = clang_Cursor_isAnonymous(Record);
138733337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        /* Recurse as long as the parent is a CXType_Record and the Record
138833337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar           is anonymous */
138933337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar      } while ( clang_getCursorType(Parent).kind == CXType_Record &&
139033337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar                RecordIsAnonymous > 0);
1391411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      {
139233337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        long long Offset = clang_Type_getOffsetOf(clang_getCursorType(Record),
1393411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis                                                  FieldName);
139433337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        long long Offset2 = clang_Cursor_getOffsetOfField(cursor);
139533337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        if (Offset == Offset2){
139633337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar            printf(" [offsetof=%lld]", Offset);
139733337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        } else {
139833337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar            /* Offsets will be different in anonymous records. */
139933337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar            printf(" [offsetof=%lld/%lld]", Offset, Offset2);
140033337ca4d89605025818daf83390ab4271d598d9Pirama Arumuga Nainar        }
1401411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      }
1402411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    }
14036bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    clang_disposeString(FieldSpelling);
1404411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  }
1405411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  /* Print if its a bitfield */
1406411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  {
1407411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    int IsBitfield = clang_Cursor_isBitField(cursor);
1408411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    if (IsBitfield)
1409411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      printf(" [BitFieldSize=%d]", clang_getFieldDeclBitWidth(cursor));
1410411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  }
1411411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  printf("\n");
1412411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  return CXChildVisit_Recurse;
1413411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis}
1414411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis
14151eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko/******************************************************************************/
1416176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines/* Mangling testing.                                                          */
1417176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines/******************************************************************************/
1418176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines
1419176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hinesstatic enum CXChildVisitResult PrintMangledName(CXCursor cursor, CXCursor p,
1420176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines                                                CXClientData d) {
1421176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  CXString MangledName;
1422176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  PrintCursor(cursor, NULL);
1423176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  MangledName = clang_Cursor_getMangling(cursor);
1424176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  printf(" [mangled=%s]\n", clang_getCString(MangledName));
1425176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  clang_disposeString(MangledName);
1426176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  return CXChildVisit_Continue;
1427176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines}
1428176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines
1429176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines/******************************************************************************/
14301eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko/* Bitwidth testing.                                                          */
14311eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko/******************************************************************************/
14321eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko
14331eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenkostatic enum CXChildVisitResult PrintBitWidth(CXCursor cursor, CXCursor p,
14341eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko                                             CXClientData d) {
143502c1b863440b067f3d674bde237146ef54c17875NAKAMURA Takumi  int Bitwidth;
14361eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  if (clang_getCursorKind(cursor) != CXCursor_FieldDecl)
14371eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    return CXChildVisit_Recurse;
14381eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko
143902c1b863440b067f3d674bde237146ef54c17875NAKAMURA Takumi  Bitwidth = clang_getFieldDeclBitWidth(cursor);
14401eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  if (Bitwidth >= 0) {
14411eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    PrintCursor(cursor, NULL);
14421eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    printf(" bitwidth=%d\n", Bitwidth);
14431eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  }
14441eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko
14451eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  return CXChildVisit_Recurse;
14461eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko}
14478e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
14488e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/******************************************************************************/
14497d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/* Loading ASTs/source.                                                       */
14507d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
14517d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
1452625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbarstatic int perform_test_load(CXIndex Idx, CXTranslationUnit TU,
145398271567b6f10be85bcc75d0bbb67416c8cfb83aTed Kremenek                             const char *filter, const char *prefix,
1454ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             CXCursorVisitor Visitor,
1455f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             PostVisitTU PV,
1456f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             const char *CommentSchemaFile) {
1457e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1458fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek  if (prefix)
1459e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    FileCheckPrefix = prefix;
1460e3ee02a324559829142f387f75daf3da118442beTed Kremenek
1461e3ee02a324559829142f387f75daf3da118442beTed Kremenek  if (Visitor) {
1462e3ee02a324559829142f387f75daf3da118442beTed Kremenek    enum CXCursorKind K = CXCursor_NotImplemented;
1463e3ee02a324559829142f387f75daf3da118442beTed Kremenek    enum CXCursorKind *ck = &K;
1464e3ee02a324559829142f387f75daf3da118442beTed Kremenek    VisitorData Data;
1465e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1466e3ee02a324559829142f387f75daf3da118442beTed Kremenek    /* Perform some simple filtering. */
1467e3ee02a324559829142f387f75daf3da118442beTed Kremenek    if (!strcmp(filter, "all") || !strcmp(filter, "local")) ck = NULL;
1468358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    else if (!strcmp(filter, "all-display") ||
1469358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor             !strcmp(filter, "local-display")) {
1470358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      ck = NULL;
1471358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      want_display_name = 1;
1472358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
1473b1ffee6e574d57b79ce2defd1b57052649221c03Daniel Dunbar    else if (!strcmp(filter, "none")) K = (enum CXCursorKind) ~0;
1474e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "category")) K = CXCursor_ObjCCategoryDecl;
1475e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "interface")) K = CXCursor_ObjCInterfaceDecl;
1476e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "protocol")) K = CXCursor_ObjCProtocolDecl;
1477e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "function")) K = CXCursor_FunctionDecl;
1478e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "typedef")) K = CXCursor_TypedefDecl;
1479e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "scan-function")) Visitor = FunctionScanVisitor;
1480e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else {
1481e3ee02a324559829142f387f75daf3da118442beTed Kremenek      fprintf(stderr, "Unknown filter for -test-load-tu: %s\n", filter);
1482e3ee02a324559829142f387f75daf3da118442beTed Kremenek      return 1;
1483e3ee02a324559829142f387f75daf3da118442beTed Kremenek    }
1484e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1485e3ee02a324559829142f387f75daf3da118442beTed Kremenek    Data.TU = TU;
1486e3ee02a324559829142f387f75daf3da118442beTed Kremenek    Data.Filter = ck;
14876bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    Data.CommentSchemaFile = CommentSchemaFile;
1488e3ee02a324559829142f387f75daf3da118442beTed Kremenek    clang_visitChildren(clang_getTranslationUnitCursor(TU), Visitor, &Data);
1489e3ee02a324559829142f387f75daf3da118442beTed Kremenek  }
1490e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1491ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek  if (PV)
1492ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek    PV(TU);
1493e3ee02a324559829142f387f75daf3da118442beTed Kremenek
1494a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
149516ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
149616ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis    clang_disposeTranslationUnit(TU);
149716ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis    return -1;
149816ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis  }
149916ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis
15000d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek  clang_disposeTranslationUnit(TU);
15010d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek  return 0;
15020d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek}
15030d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
1504fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenekint perform_test_load_tu(const char *file, const char *filter,
1505ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                         const char *prefix, CXCursorVisitor Visitor,
1506ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                         PostVisitTU PV) {
1507625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXIndex Idx;
1508625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXTranslationUnit TU;
1509020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  int result;
1510e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  Idx = clang_createIndex(/* excludeDeclsFromPCH */
15110a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                          !strcmp(filter, "local") ? 1 : 0,
1512fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                          /* displayDiagnostics=*/1);
1513e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1514020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  if (!CreateTranslationUnit(Idx, file, &TU)) {
1515020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
1516625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    return 1;
1517020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  }
1518625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1519f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, prefix, Visitor, PV, NULL);
1520020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  clang_disposeIndex(Idx);
1521020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  return result;
1522625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar}
1523625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1524ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenekint perform_test_load_source(int argc, const char **argv,
1525ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             const char *filter, CXCursorVisitor Visitor,
1526ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             PostVisitTU PV) {
1527ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  CXIndex Idx;
1528ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  CXTranslationUnit TU;
1529f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentSchemaFile;
15304db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
15314db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int num_unsaved_files = 0;
1532651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
15334db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int result;
1534651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
1535ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  Idx = clang_createIndex(/* excludeDeclsFromPCH */
1536358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                          (!strcmp(filter, "local") ||
1537358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                           !strcmp(filter, "local-display"))? 1 : 0,
1538cd6dcb3434d77b1f652416f2b5c3c62394f1472cArgyrios Kyrtzidis                          /* displayDiagnostics=*/1);
1539ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1540f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if ((CommentSchemaFile = parse_comments_schema(argc, argv))) {
1541f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    argc--;
1542f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    argv++;
1543f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
1544f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
1545020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
1546020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
15474db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    return -1;
1548020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  }
15494db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
1550651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(Idx, 0,
1551651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + num_unsaved_files,
1552651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - num_unsaved_files,
1553651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    unsaved_files, num_unsaved_files,
1554651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    getDefaultParsingOptions(), &TU);
1555651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
1556ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar    fprintf(stderr, "Unable to load translation unit!\n");
1557651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
1558abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
1559020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
1560ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar    return 1;
1561ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  }
1562ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1563f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, NULL, Visitor, PV,
1564f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             CommentSchemaFile);
15654db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1566020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  clang_disposeIndex(Idx);
15674db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  return result;
1568ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar}
1569ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1570abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregorint perform_test_reparse_source(int argc, const char **argv, int trials,
1571abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                const char *filter, CXCursorVisitor Visitor,
1572abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                PostVisitTU PV) {
1573abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  CXIndex Idx;
1574abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  CXTranslationUnit TU;
1575abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
1576abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int num_unsaved_files = 0;
1577651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int compiler_arg_idx = 0;
1578651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
1579651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int result, i;
1580abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int trial;
158140098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  int remap_after_trial = 0;
158240098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  char *endptr = 0;
1583abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1584abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  Idx = clang_createIndex(/* excludeDeclsFromPCH */
1585abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                          !strcmp(filter, "local") ? 1 : 0,
1586cd6dcb3434d77b1f652416f2b5c3c62394f1472cArgyrios Kyrtzidis                          /* displayDiagnostics=*/1);
1587abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1588abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
1589abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    clang_disposeIndex(Idx);
1590abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    return -1;
1591abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1592651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
1593651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  for (i = 0; i < argc; ++i) {
1594651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (strcmp(argv[i], "--") == 0)
1595651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      break;
1596651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  }
1597651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (i < argc)
1598651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    compiler_arg_idx = i+1;
1599651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (num_unsaved_files > compiler_arg_idx)
1600651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    compiler_arg_idx = num_unsaved_files;
1601abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1602c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar  /* Load the initial translation unit -- we do this without honoring remapped
1603c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar   * files, so that we have a way to test results after changing the source. */
1604651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(Idx, 0,
1605651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + compiler_arg_idx,
1606651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - compiler_arg_idx,
1607651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    0, 0, getDefaultParsingOptions(), &TU);
1608651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
1609abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
1610651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
1611abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
1612abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    clang_disposeIndex(Idx);
1613abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    return 1;
1614abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1615abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1616bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
1617bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis    return -1;
1618bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
161940098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  if (getenv("CINDEXTEST_REMAP_AFTER_TRIAL")) {
162040098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis    remap_after_trial =
162140098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis        strtol(getenv("CINDEXTEST_REMAP_AFTER_TRIAL"), &endptr, 10);
162240098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  }
162340098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis
1624abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  for (trial = 0; trial < trials; ++trial) {
1625651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    free_remapped_files(unsaved_files, num_unsaved_files);
1626651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (parse_remapped_files_with_try(trial, argc, argv, 0,
1627651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                      &unsaved_files, &num_unsaved_files)) {
1628651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      clang_disposeTranslationUnit(TU);
1629651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      clang_disposeIndex(Idx);
1630651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      return -1;
1631651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    }
1632651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
1633651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    Err = clang_reparseTranslationUnit(
1634651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        TU,
1635651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        trial >= remap_after_trial ? num_unsaved_files : 0,
1636651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        trial >= remap_after_trial ? unsaved_files : 0,
1637651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        clang_defaultReparseOptions(TU));
1638651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (Err != CXError_Success) {
1639c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar      fprintf(stderr, "Unable to reparse translation unit!\n");
1640651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      describeLibclangFailure(Err);
1641abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      clang_disposeTranslationUnit(TU);
1642abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      free_remapped_files(unsaved_files, num_unsaved_files);
1643abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      clang_disposeIndex(Idx);
1644abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      return -1;
1645abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    }
1646bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
1647bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
1648bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis      return -1;
1649abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1650abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1651f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, NULL, Visitor, PV, NULL);
1652bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
1653abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1654abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  clang_disposeIndex(Idx);
1655abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  return result;
1656abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor}
1657abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
16580d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
16591c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/* Logic for testing clang_getCursor().                                       */
16601c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/******************************************************************************/
16611c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
1662dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregorstatic void print_cursor_file_scan(CXTranslationUnit TU, CXCursor cursor,
16631c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek                                   unsigned start_line, unsigned start_col,
16641d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                                   unsigned end_line, unsigned end_col,
16651d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                                   const char *prefix) {
16669096a20fc0fd799911b738ce7a9134dbacaf64b7Ted Kremenek  printf("// %s: ", FileCheckPrefix);
16671d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek  if (prefix)
16681d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    printf("-%s", prefix);
166951b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  PrintExtent(stdout, start_line, start_col, end_line, end_col);
167051b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  printf(" ");
1671f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(cursor, NULL);
16721c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  printf("\n");
16731c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
16741c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
16751d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenekstatic int perform_file_scan(const char *ast_file, const char *source_file,
16761d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                             const char *prefix) {
16771c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  CXIndex Idx;
16781c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  CXTranslationUnit TU;
16791c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  FILE *fp;
168010f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXCursor prevCursor = clang_getNullCursor();
1681b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  CXFile file;
16822389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  unsigned line = 1, col = 1;
16838f0bf81c0f3ab31881cc7db7914e6978bb2a19afDaniel Dunbar  unsigned start_line = 1, start_col = 1;
1684e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
16850a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
1686fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                                /* displayDiagnostics=*/1))) {
16871c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Could not create Index\n");
16881c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
16891c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1690e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
16911c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  if (!CreateTranslationUnit(Idx, ast_file, &TU))
16921c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
1693e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
16941c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  if ((fp = fopen(source_file, "r")) == NULL) {
16951c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Could not open '%s'\n", source_file);
1696176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines    clang_disposeTranslationUnit(TU);
16971c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
16981c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1699e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1700b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  file = clang_getFile(TU, source_file);
17012389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  for (;;) {
17022389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    CXCursor cursor;
17032389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    int c = fgetc(fp);
17042389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar
17052389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if (c == '\n') {
17062389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      ++line;
17072389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      col = 1;
17082389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    } else
17092389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      ++col;
17102389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar
17112389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    /* Check the cursor at this position, and dump the previous one if we have
17122389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar     * found something new.
17132389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar     */
17142389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    cursor = clang_getCursor(TU, clang_getLocation(TU, file, line, col));
17152389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if ((c == EOF || !clang_equalCursors(cursor, prevCursor)) &&
17162389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar        prevCursor.kind != CXCursor_InvalidFile) {
1717dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor      print_cursor_file_scan(TU, prevCursor, start_line, start_col,
1718d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar                             line, col, prefix);
17192389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      start_line = line;
17202389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      start_col = col;
1721a9933b98399f656653a0876fc39e5b9093efb732Benjamin Kramer    }
17222389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if (c == EOF)
17232389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      break;
1724a9933b98399f656653a0876fc39e5b9093efb732Benjamin Kramer
17252389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    prevCursor = cursor;
17261c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1727e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
17281c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  fclose(fp);
17294f5e21e24fb9e6ec473a13f83b5c9a2c41501a70Douglas Gregor  clang_disposeTranslationUnit(TU);
17304f5e21e24fb9e6ec473a13f83b5c9a2c41501a70Douglas Gregor  clang_disposeIndex(Idx);
17311c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  return 0;
17321c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
17331c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
17341c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/******************************************************************************/
173532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor/* Logic for testing clang code completion.                                   */
17360d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
17370d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
17380c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor/* Parse file:line:column from the input string. Returns 0 on success, non-zero
17390c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor   on failure. If successful, the pointer *filename will contain newly-allocated
17400c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor   memory (that will be owned by the caller) to store the file name. */
1741e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekint parse_file_line_column(const char *input, char **filename, unsigned *line,
1742fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                           unsigned *column, unsigned *second_line,
1743fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                           unsigned *second_column) {
174488d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor  /* Find the second colon. */
1745fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  const char *last_colon = strrchr(input, ':');
1746fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned values[4], i;
1747fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned num_values = (second_line && second_column)? 4 : 2;
1748fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
17490c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  char *endptr = 0;
1750fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!last_colon || last_colon == input) {
1751fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (num_values == 4)
1752fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      fprintf(stderr, "could not parse filename:line:column:line:column in "
1753fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              "'%s'\n", input);
1754fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    else
1755fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      fprintf(stderr, "could not parse filename:line:column in '%s'\n", input);
17560c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor    return 1;
17570c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
17580c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1759fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  for (i = 0; i != num_values; ++i) {
1760fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *prev_colon;
176188d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor
1762fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    /* Parse the next line or column. */
1763fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    values[num_values - i - 1] = strtol(last_colon + 1, &endptr, 10);
1764fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (*endptr != 0 && *endptr != ':') {
1765e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr, "could not parse %s in '%s'\n",
1766fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              (i % 2 ? "column" : "line"), input);
1767fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      return 1;
1768fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
1769e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1770fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (i + 1 == num_values)
1771fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      break;
177288d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor
1773fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    /* Find the previous colon. */
1774fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    prev_colon = last_colon - 1;
1775fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    while (prev_colon != input && *prev_colon != ':')
1776fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      --prev_colon;
1777fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (prev_colon == input) {
1778e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr, "could not parse %s in '%s'\n",
1779fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              (i % 2 == 0? "column" : "line"), input);
1780e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return 1;
1781fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
1782fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
1783fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    last_colon = prev_colon;
17840c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
1785fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
1786fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *line = values[0];
1787fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *column = values[1];
1788e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1789fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (second_line && second_column) {
1790fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *second_line = values[2];
1791fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *second_column = values[3];
1792fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
1793fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
179488d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor  /* Copy the file name. */
1795fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *filename = (char*)malloc(last_colon - input + 1);
1796fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  memcpy(*filename, input, last_colon - input);
1797fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  (*filename)[last_colon - input] = 0;
17980c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  return 0;
17990c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
18000c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
18010c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregorconst char *
18020c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregorclang_getCompletionChunkKindSpelling(enum CXCompletionChunkKind Kind) {
18030c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  switch (Kind) {
18040c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Optional: return "Optional";
18050c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_TypedText: return "TypedText";
18060c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Text: return "Text";
18070c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Placeholder: return "Placeholder";
18080c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Informative: return "Informative";
18090c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_CurrentParameter: return "CurrentParameter";
18100c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftParen: return "LeftParen";
18110c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightParen: return "RightParen";
18120c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftBracket: return "LeftBracket";
18130c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightBracket: return "RightBracket";
18140c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftBrace: return "LeftBrace";
18150c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightBrace: return "RightBrace";
18160c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftAngle: return "LeftAngle";
18170c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightAngle: return "RightAngle";
18180c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Comma: return "Comma";
1819ff5ce6eefc7c253ef6edf4d4bfc996fdd82d09aaDouglas Gregor  case CXCompletionChunk_ResultType: return "ResultType";
182001dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_Colon: return "Colon";
182101dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_SemiColon: return "SemiColon";
182201dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_Equal: return "Equal";
182301dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_HorizontalSpace: return "HorizontalSpace";
182401dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_VerticalSpace: return "VerticalSpace";
18250c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
1826e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
18270c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  return "Unknown";
18280c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
18290c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1830dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidisstatic int checkForErrors(CXTranslationUnit TU) {
1831dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  unsigned Num, i;
1832dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  CXDiagnostic Diag;
1833dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  CXString DiagStr;
1834dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1835dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (!getenv("CINDEXTEST_FAILONERROR"))
1836dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return 0;
1837dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1838dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  Num = clang_getNumDiagnostics(TU);
1839dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  for (i = 0; i != Num; ++i) {
1840dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    Diag = clang_getDiagnostic(TU, i);
1841dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (clang_getDiagnosticSeverity(Diag) >= CXDiagnostic_Error) {
1842dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      DiagStr = clang_formatDiagnostic(Diag,
1843dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis                                       clang_defaultDiagnosticDisplayOptions());
1844dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      fprintf(stderr, "%s\n", clang_getCString(DiagStr));
1845dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      clang_disposeString(DiagStr);
1846dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      clang_disposeDiagnostic(Diag);
1847dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
1848dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    }
1849dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    clang_disposeDiagnostic(Diag);
1850dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
1851dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1852dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  return 0;
1853dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis}
1854dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
18556bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic void print_completion_string(CXCompletionString completion_string,
18566bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                    FILE *file) {
1857f8297f1512d29c88f20bf2901f04cc04182a3eeeDaniel Dunbar  int I, N;
1858e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
18593ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  N = clang_getNumCompletionChunks(completion_string);
18600c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  for (I = 0; I != N; ++I) {
18612ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    CXString text;
18622ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    const char *cstr;
18630c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor    enum CXCompletionChunkKind Kind
18643ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      = clang_getCompletionChunkKind(completion_string, I);
1865e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
18663ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor    if (Kind == CXCompletionChunk_Optional) {
18673ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      fprintf(file, "{Optional ");
18683ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      print_completion_string(
1869e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                clang_getCompletionChunkCompletionString(completion_string, I),
18703ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor                              file);
18713ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      fprintf(file, "}");
18723ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      continue;
18735a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor    }
18745a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor
18755a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor    if (Kind == CXCompletionChunk_VerticalSpace) {
18765a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor      fprintf(file, "{VerticalSpace  }");
18775a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor      continue;
18783ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor    }
1879e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1880d5a2089663d81faee0de031deabc418fa53ecf3bDouglas Gregor    text = clang_getCompletionChunkText(completion_string, I);
18812ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    cstr = clang_getCString(text);
1882e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(file, "{%s %s}",
18830c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor            clang_getCompletionChunkKindSpelling(Kind),
18842ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek            cstr ? cstr : "");
18852ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    clang_disposeString(text);
18860c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
18872ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek
18883ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor}
18893ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor
18906bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic void print_completion_result(CXCompletionResult *completion_result,
18916bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                    FILE *file) {
189210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXString ks = clang_getCursorKindSpelling(completion_result->CursorKind);
18936164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  unsigned annotationCount;
1894ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  enum CXCursorKind ParentKind;
1895ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  CXString ParentName;
1896d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  CXString BriefComment;
1897d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  const char *BriefCommentCString;
1898ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor
1899e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  fprintf(file, "%s:", clang_getCString(ks));
1900e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  clang_disposeString(ks);
1901e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
19023ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  print_completion_string(completion_result->CompletionString, file);
190358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  fprintf(file, " (%u)",
190412e131385e892e3723483a1081a89bcad29c8a84Douglas Gregor          clang_getCompletionPriority(completion_result->CompletionString));
190558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  switch (clang_getCompletionAvailability(completion_result->CompletionString)){
190658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_Available:
190758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
190858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
190958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_Deprecated:
191058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    fprintf(file, " (deprecated)");
191158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
191258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
191358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_NotAvailable:
191458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    fprintf(file, " (unavailable)");
191558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
1916d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen
1917d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen  case CXAvailability_NotAccessible:
1918d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen    fprintf(file, " (inaccessible)");
1919d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen    break;
192058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  }
19216164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen
19226164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  annotationCount = clang_getCompletionNumAnnotations(
19236164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen        completion_result->CompletionString);
19246164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  if (annotationCount) {
19256164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    unsigned i;
19266164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    fprintf(file, " (");
19276164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    for (i = 0; i < annotationCount; ++i) {
19286164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen      if (i != 0)
19296164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen        fprintf(file, ", ");
19306164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen      fprintf(file, "\"%s\"",
19316164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen              clang_getCString(clang_getCompletionAnnotation(
19326164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen                                 completion_result->CompletionString, i)));
19336164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    }
19346164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    fprintf(file, ")");
19356164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  }
19366164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen
1937ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  if (!getenv("CINDEXTEST_NO_COMPLETION_PARENTS")) {
1938ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    ParentName = clang_getCompletionParent(completion_result->CompletionString,
1939ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor                                           &ParentKind);
1940ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    if (ParentKind != CXCursor_NotImplemented) {
194110f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXString KindSpelling = clang_getCursorKindSpelling(ParentKind);
1942ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      fprintf(file, " (parent: %s '%s')",
1943ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor              clang_getCString(KindSpelling),
1944ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor              clang_getCString(ParentName));
1945ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      clang_disposeString(KindSpelling);
1946ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    }
1947ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    clang_disposeString(ParentName);
1948ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  }
1949d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko
1950d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  BriefComment = clang_getCompletionBriefComment(
1951d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko                                        completion_result->CompletionString);
1952d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  BriefCommentCString = clang_getCString(BriefComment);
1953d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (BriefCommentCString && *BriefCommentCString != '\0') {
1954d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    fprintf(file, "(brief comment: %s)", BriefCommentCString);
1955d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  }
1956d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  clang_disposeString(BriefComment);
1957ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor
195858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  fprintf(file, "\n");
19590c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
19600c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
19613da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregorvoid print_completion_contexts(unsigned long long contexts, FILE *file) {
19623da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  fprintf(file, "Completion contexts:\n");
19633da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts == CXCompletionContext_Unknown) {
19643da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Unknown\n");
19653da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19663da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_AnyType) {
19673da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Any type\n");
19683da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19693da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_AnyValue) {
19703da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Any value\n");
19713da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19723da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCObjectValue) {
19733da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C object value\n");
19743da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19753da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCSelectorValue) {
19763da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C selector value\n");
19773da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19783da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_CXXClassTypeValue) {
19793da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "C++ class type value\n");
19803da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19813da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_DotMemberAccess) {
19823da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Dot member access\n");
19833da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19843da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ArrowMemberAccess) {
19853da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Arrow member access\n");
19863da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19873da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCPropertyAccess) {
19883da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C property access\n");
19893da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19903da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_EnumTag) {
19913da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Enum tag\n");
19923da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19933da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_UnionTag) {
19943da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Union tag\n");
19953da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19963da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_StructTag) {
19973da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Struct tag\n");
19983da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19993da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ClassTag) {
20003da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Class name\n");
20013da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20023da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_Namespace) {
20033da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Namespace or namespace alias\n");
20043da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20053da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_NestedNameSpecifier) {
20063da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Nested name specifier\n");
20073da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20083da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCInterface) {
20093da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C interface\n");
20103da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20113da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCProtocol) {
20123da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C protocol\n");
20133da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20143da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCCategory) {
20153da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C category\n");
20163da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20173da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCInstanceMessage) {
20183da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C instance method\n");
20193da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20203da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCClassMessage) {
20213da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C class method\n");
20223da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20233da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCSelectorName) {
20243da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C selector name\n");
20253da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20263da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_MacroName) {
20273da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Macro name\n");
20283da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20293da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_NaturalLanguage) {
20303da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Natural language\n");
20313da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
20323da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor}
20333da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
20341e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregorint my_stricmp(const char *s1, const char *s2) {
20351e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  while (*s1 && *s2) {
20366d5552131a4637f9bbe8c93386648e9bbb2c30feNAKAMURA Takumi    int c1 = tolower((unsigned char)*s1), c2 = tolower((unsigned char)*s2);
20371e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    if (c1 < c2)
20381e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      return -1;
20391e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    else if (c1 > c2)
20401e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      return 1;
20411e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
20421e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    ++s1;
20431e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    ++s2;
20441e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  }
20451e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
20461e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  if (*s1)
20471e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    return 1;
20481e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  else if (*s2)
20491e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    return -1;
20501e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  return 0;
20511e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor}
20521e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
20531982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregorint perform_code_completion(int argc, const char **argv, int timing_only) {
20540c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  const char *input = argv[1];
20550c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  char *filename = 0;
20560c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  unsigned line;
20570c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  unsigned column;
2058f8297f1512d29c88f20bf2901f04cc04182a3eeeDaniel Dunbar  CXIndex CIdx;
2059f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  int errorCode;
2060735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
2061735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  int num_unsaved_files = 0;
2062ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  CXCodeCompleteResults *results = 0;
2063651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
2064651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  CXTranslationUnit TU;
206532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  unsigned I, Repeats = 1;
206632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  unsigned completionOptions = clang_defaultCodeCompleteOptions();
206732be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
206832be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (getenv("CINDEXTEST_CODE_COMPLETE_PATTERNS"))
206932be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    completionOptions |= CXCodeComplete_IncludeCodePatterns;
2070d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (getenv("CINDEXTEST_COMPLETION_BRIEF_COMMENTS"))
2071d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    completionOptions |= CXCodeComplete_IncludeBriefComments;
2072df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor
20731982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  if (timing_only)
20741982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    input += strlen("-code-completion-timing=");
20751982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  else
20761982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    input += strlen("-code-completion-at=");
20771982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor
2078e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if ((errorCode = parse_file_line_column(input, &filename, &line, &column,
2079fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                          0, 0)))
2080f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek    return errorCode;
20810c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
2082735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  if (parse_remapped_files(argc, argv, 2, &unsaved_files, &num_unsaved_files))
2083735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor    return -1;
2084735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor
208532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  CIdx = clang_createIndex(0, 0);
208632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
208732be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (getenv("CINDEXTEST_EDITING"))
208832be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    Repeats = 5;
2089651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
2090651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(CIdx, 0,
2091651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + num_unsaved_files + 2,
2092651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - num_unsaved_files - 2,
2093651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    0, 0, getDefaultParsingOptions(), &TU);
2094651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
209532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
2096651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
209732be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    return 1;
209832be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  }
209908bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor
2100651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_reparseTranslationUnit(TU, 0, 0,
2101651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                     clang_defaultReparseOptions(TU));
2102651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
2103651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
210408bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor    fprintf(stderr, "Unable to reparse translation init!\n");
2105651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
2106651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_disposeTranslationUnit(TU);
210708bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor    return 1;
210808bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor  }
2109651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
211032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  for (I = 0; I != Repeats; ++I) {
211132be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    results = clang_codeCompleteAt(TU, filename, line, column,
211232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                   unsaved_files, num_unsaved_files,
211332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                   completionOptions);
211432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    if (!results) {
211532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor      fprintf(stderr, "Unable to perform code completion!\n");
21162de41c9c99e0e9ae6488d04c08423a5c1190109eDaniel Dunbar      return 1;
21172de41c9c99e0e9ae6488d04c08423a5c1190109eDaniel Dunbar    }
211832be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    if (I != Repeats-1)
211932be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor      clang_disposeCodeCompleteResults(results);
212032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  }
2121936ea3b590117d2cd73b1b92621d06c4a7edbe60Douglas Gregor
2122ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  if (results) {
2123e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    unsigned i, n = results->NumResults, containerIsIncomplete = 0;
21243da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    unsigned long long contexts;
2125e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    enum CXCursorKind containerKind;
21260a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    CXString objCSelector;
21270a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    const char *selectorString;
21281e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    if (!timing_only) {
21291e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      /* Sort the code-completion results based on the typed text. */
21301e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      clang_sortCodeCompletionResults(results->Results, results->NumResults);
21311e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
21321982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor      for (i = 0; i != n; ++i)
21331982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor        print_completion_result(results->Results + i, stdout);
21341e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    }
2135a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    n = clang_codeCompleteGetNumDiagnostics(results);
2136a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    for (i = 0; i != n; ++i) {
2137a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      CXDiagnostic diag = clang_codeCompleteGetDiagnostic(results, i);
2138a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      PrintDiagnostic(diag);
2139a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      clang_disposeDiagnostic(diag);
2140a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    }
21413da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
21423da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    contexts = clang_codeCompleteGetContexts(results);
21433da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    print_completion_contexts(contexts, stdout);
21443da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
21450a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    containerKind = clang_codeCompleteGetContainerKind(results,
21460a47d69af8bda945352997af3da4687a3356096aDouglas Gregor                                                       &containerIsIncomplete);
2147e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
2148e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    if (containerKind != CXCursor_InvalidCode) {
2149e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      /* We have found a container */
2150e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      CXString containerUSR, containerKindSpelling;
2151e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      containerKindSpelling = clang_getCursorKindSpelling(containerKind);
2152e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      printf("Container Kind: %s\n", clang_getCString(containerKindSpelling));
2153e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      clang_disposeString(containerKindSpelling);
2154e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
2155e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      if (containerIsIncomplete) {
2156e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor        printf("Container is incomplete\n");
2157e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      }
2158e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      else {
2159e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor        printf("Container is complete\n");
2160e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      }
2161e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
2162e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      containerUSR = clang_codeCompleteGetContainerUSR(results);
2163e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      printf("Container USR: %s\n", clang_getCString(containerUSR));
2164e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      clang_disposeString(containerUSR);
2165e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    }
2166e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
21670a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    objCSelector = clang_codeCompleteGetObjCSelector(results);
21680a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    selectorString = clang_getCString(objCSelector);
21690a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    if (selectorString && strlen(selectorString) > 0) {
21700a47d69af8bda945352997af3da4687a3356096aDouglas Gregor      printf("Objective-C selector: %s\n", selectorString);
21710a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    }
21720a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    clang_disposeString(objCSelector);
21730a47d69af8bda945352997af3da4687a3356096aDouglas Gregor
2174ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor    clang_disposeCodeCompleteResults(results);
2175ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  }
2176df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor  clang_disposeTranslationUnit(TU);
21770c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  clang_disposeIndex(CIdx);
21780c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  free(filename);
2179e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2180735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
2181735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor
2182f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  return 0;
21830c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
21840c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
2185f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregortypedef struct {
2186f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  char *filename;
2187f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned line;
2188f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned column;
2189f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor} CursorSourceLocation;
2190f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor
2191aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic int inspect_cursor_at(int argc, const char **argv) {
2192f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXIndex CIdx;
2193f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  int errorCode;
2194f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
2195f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  int num_unsaved_files = 0;
2196651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
2197f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXTranslationUnit TU;
2198f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXCursor Cursor;
2199f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CursorSourceLocation *Locations = 0;
2200f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned NumLocations = 0, Loc;
22018e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  unsigned Repeats = 1;
2202bdc4b366e80c125184a3b3c56fa4619cb4ac9e45Douglas Gregor  unsigned I;
22038e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
2204e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  /* Count the number of locations. */
2205f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  while (strstr(argv[NumLocations+1], "-cursor-at=") == argv[NumLocations+1])
2206f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    ++NumLocations;
2207e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2208f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  /* Parse the locations. */
2209f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  assert(NumLocations > 0 && "Unable to count locations?");
2210f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  Locations = (CursorSourceLocation *)malloc(
2211f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                                  NumLocations * sizeof(CursorSourceLocation));
2212f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  for (Loc = 0; Loc < NumLocations; ++Loc) {
2213f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    const char *input = argv[Loc + 1] + strlen("-cursor-at=");
2214e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    if ((errorCode = parse_file_line_column(input, &Locations[Loc].filename,
2215e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                                            &Locations[Loc].line,
2216fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                            &Locations[Loc].column, 0, 0)))
2217f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor      return errorCode;
2218f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
2219e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2220e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (parse_remapped_files(argc, argv, NumLocations + 1, &unsaved_files,
2221f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                           &num_unsaved_files))
2222f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return -1;
2223e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
22248e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  if (getenv("CINDEXTEST_EDITING"))
22258e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    Repeats = 5;
22268e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
22278e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  /* Parse the translation unit. When we're testing clang_getCursor() after
22288e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor     reparsing, don't remap unsaved files until the second parse. */
22298e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  CIdx = clang_createIndex(1, 1);
2230651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(CIdx, argv[argc - 1],
2231651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   argv + num_unsaved_files + 1 + NumLocations,
2232651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   argc - num_unsaved_files - 2 - NumLocations,
2233651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   unsaved_files,
2234651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   Repeats > 1? 0 : num_unsaved_files,
2235651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   getDefaultParsingOptions(), &TU);
2236651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
2237f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    fprintf(stderr, "unable to parse input\n");
2238651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
2239f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return -1;
2240f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
2241e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2242dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
2243dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return -1;
2244dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2245bdc4b366e80c125184a3b3c56fa4619cb4ac9e45Douglas Gregor  for (I = 0; I != Repeats; ++I) {
2246651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (Repeats > 1) {
2247651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Err = clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2248651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         clang_defaultReparseOptions(TU));
2249651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (Err != CXError_Success) {
2250651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        describeLibclangFailure(Err);
2251651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        clang_disposeTranslationUnit(TU);
2252651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        return 1;
2253651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      }
22548e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    }
2255dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2256dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
2257dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
22588e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
22598e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    for (Loc = 0; Loc < NumLocations; ++Loc) {
22608e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      CXFile file = clang_getFile(TU, Locations[Loc].filename);
22618e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      if (!file)
22628e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        continue;
22638e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
22648e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      Cursor = clang_getCursor(TU,
22658e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                               clang_getLocation(TU, file, Locations[Loc].line,
22668e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                                 Locations[Loc].column));
2267dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2268dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      if (checkForErrors(TU) != 0)
2269dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        return -1;
2270dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
22718e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      if (I + 1 == Repeats) {
227210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        CXCompletionString completionString = clang_getCursorCompletionString(
227310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella                                                                        Cursor);
227410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        CXSourceLocation CursorLoc = clang_getCursorLocation(Cursor);
227566373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        CXString Spelling;
227666373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        const char *cspell;
227766373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        unsigned line, column;
227866373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        clang_getSpellingLocation(CursorLoc, 0, &line, &column, 0);
227966373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        printf("%d:%d ", line, column);
2280f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCursor(Cursor, NULL);
228166373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        PrintCursorExtent(Cursor);
228266373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        Spelling = clang_getCursorSpelling(Cursor);
228366373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        cspell = clang_getCString(Spelling);
2284ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis        if (cspell && strlen(cspell) != 0) {
2285ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          unsigned pieceIndex;
2286ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          printf(" Spelling=%s (", cspell);
2287ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          for (pieceIndex = 0; ; ++pieceIndex) {
228810f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella            CXSourceRange range =
228910f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella              clang_Cursor_getSpellingNameRange(Cursor, pieceIndex, 0);
2290ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis            if (clang_Range_isNull(range))
2291ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis              break;
2292ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis            PrintRange(range, 0);
2293ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          }
2294ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          printf(")");
2295ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis        }
229666373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        clang_disposeString(Spelling);
229734ebe1e1b0779bcea2f277bc6b4e9dd98bf70b7bArgyrios Kyrtzidis        if (clang_Cursor_getObjCSelectorIndex(Cursor) != -1)
22986bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines          printf(" Selector index=%d",
22996bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                 clang_Cursor_getObjCSelectorIndex(Cursor));
2300f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis        if (clang_Cursor_isDynamicCall(Cursor))
2301f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis          printf(" Dynamic-call");
2302e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis        if (Cursor.kind == CXCursor_ObjCMessageExpr) {
230310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella          CXType T = clang_Cursor_getReceiverType(Cursor);
230410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella          CXString S = clang_getTypeKindSpelling(T.kind);
2305e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis          printf(" Receiver-type=%s", clang_getCString(S));
2306e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis          clang_disposeString(S);
2307e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis        }
2308f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis
23095d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis        {
23105d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis          CXModule mod = clang_Cursor_getModule(Cursor);
2311e858e667c14ce4a9df5a4bbae770a0a3a3c8723eArgyrios Kyrtzidis          CXFile astFile;
2312e858e667c14ce4a9df5a4bbae770a0a3a3c8723eArgyrios Kyrtzidis          CXString name, astFilename;
23135d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis          unsigned i, numHeaders;
23145d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis          if (mod) {
2315e858e667c14ce4a9df5a4bbae770a0a3a3c8723eArgyrios Kyrtzidis            astFile = clang_Module_getASTFile(mod);
2316e858e667c14ce4a9df5a4bbae770a0a3a3c8723eArgyrios Kyrtzidis            astFilename = clang_getFileName(astFile);
23175d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            name = clang_Module_getFullName(mod);
2318c1d22393628a145e54396c0ac66e9625d13a7658Argyrios Kyrtzidis            numHeaders = clang_Module_getNumTopLevelHeaders(TU, mod);
23196bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines            printf(" ModuleName=%s (%s) system=%d Headers(%d):",
2320e858e667c14ce4a9df5a4bbae770a0a3a3c8723eArgyrios Kyrtzidis                   clang_getCString(name), clang_getCString(astFilename),
23216bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                   clang_Module_isSystem(mod), numHeaders);
23225d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            clang_disposeString(name);
2323e858e667c14ce4a9df5a4bbae770a0a3a3c8723eArgyrios Kyrtzidis            clang_disposeString(astFilename);
23245d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            for (i = 0; i < numHeaders; ++i) {
232510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella              CXFile file = clang_Module_getTopLevelHeader(TU, mod, i);
232610f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella              CXString filename = clang_getFileName(file);
23275d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis              printf("\n%s", clang_getCString(filename));
23285d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis              clang_disposeString(filename);
23295d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            }
23305d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis          }
23315d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis        }
23325d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis
23338fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor        if (completionString != NULL) {
23348fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor          printf("\nCompletion string: ");
23358fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor          print_completion_string(completionString, stdout);
23368fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor        }
23378e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        printf("\n");
23388e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        free(Locations[Loc].filename);
23398e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      }
23408e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    }
2341f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
23428e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
2343a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
2344f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  clang_disposeTranslationUnit(TU);
2345f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  clang_disposeIndex(CIdx);
2346f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  free(Locations);
2347f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
2348f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  return 0;
2349f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor}
2350f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor
2351aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic enum CXVisitorResult findFileRefsVisit(void *context,
2352aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                         CXCursor cursor, CXSourceRange range) {
2353aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (clang_Range_isNull(range))
2354aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return CXVisit_Continue;
2355aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2356f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(cursor, NULL);
2357aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintRange(range, "");
2358aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  printf("\n");
2359aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  return CXVisit_Continue;
2360aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis}
2361aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2362aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic int find_file_refs_at(int argc, const char **argv) {
2363aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXIndex CIdx;
2364aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  int errorCode;
2365aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  struct CXUnsavedFile *unsaved_files = 0;
2366aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  int num_unsaved_files = 0;
2367651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
2368aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXTranslationUnit TU;
2369aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXCursor Cursor;
2370aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CursorSourceLocation *Locations = 0;
2371aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned NumLocations = 0, Loc;
2372aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned Repeats = 1;
2373aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned I;
2374aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2375aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Count the number of locations. */
2376aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  while (strstr(argv[NumLocations+1], "-file-refs-at=") == argv[NumLocations+1])
2377aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    ++NumLocations;
2378aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2379aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Parse the locations. */
2380aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  assert(NumLocations > 0 && "Unable to count locations?");
2381aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  Locations = (CursorSourceLocation *)malloc(
2382aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  NumLocations * sizeof(CursorSourceLocation));
2383aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  for (Loc = 0; Loc < NumLocations; ++Loc) {
2384aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    const char *input = argv[Loc + 1] + strlen("-file-refs-at=");
2385aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if ((errorCode = parse_file_line_column(input, &Locations[Loc].filename,
2386aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                            &Locations[Loc].line,
2387aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                            &Locations[Loc].column, 0, 0)))
2388aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return errorCode;
2389aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2390aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2391aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (parse_remapped_files(argc, argv, NumLocations + 1, &unsaved_files,
2392aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                           &num_unsaved_files))
2393aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return -1;
2394aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2395aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING"))
2396aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    Repeats = 5;
2397aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2398aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Parse the translation unit. When we're testing clang_getCursor() after
2399aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis     reparsing, don't remap unsaved files until the second parse. */
2400aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CIdx = clang_createIndex(1, 1);
2401651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(CIdx, argv[argc - 1],
2402651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + num_unsaved_files + 1 + NumLocations,
2403651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - num_unsaved_files - 2 - NumLocations,
2404651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    unsaved_files,
2405651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    Repeats > 1? 0 : num_unsaved_files,
2406651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    getDefaultParsingOptions(), &TU);
2407651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
2408aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    fprintf(stderr, "unable to parse input\n");
2409651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
2410651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_disposeTranslationUnit(TU);
2411aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return -1;
2412aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2413aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2414dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
2415dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return -1;
2416dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2417aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  for (I = 0; I != Repeats; ++I) {
2418651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (Repeats > 1) {
2419651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Err = clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2420651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         clang_defaultReparseOptions(TU));
2421651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (Err != CXError_Success) {
2422651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        describeLibclangFailure(Err);
2423651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        clang_disposeTranslationUnit(TU);
2424651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        return 1;
2425651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      }
2426aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
2427dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2428dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
2429dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
2430aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2431aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    for (Loc = 0; Loc < NumLocations; ++Loc) {
2432aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      CXFile file = clang_getFile(TU, Locations[Loc].filename);
2433aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (!file)
2434aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        continue;
2435aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2436aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      Cursor = clang_getCursor(TU,
2437aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                               clang_getLocation(TU, file, Locations[Loc].line,
2438aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                                 Locations[Loc].column));
2439dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2440dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      if (checkForErrors(TU) != 0)
2441dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        return -1;
2442dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2443aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (I + 1 == Repeats) {
244426fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen        CXCursorAndRangeVisitor visitor = { 0, findFileRefsVisit };
2445f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCursor(Cursor, NULL);
2446aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        printf("\n");
2447aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        clang_findReferencesInFile(Cursor, file, visitor);
2448aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        free(Locations[Loc].filename);
2449dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2450dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        if (checkForErrors(TU) != 0)
2451dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis          return -1;
2452aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      }
2453aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
2454aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2455aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2456aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintDiagnostics(TU);
2457aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  clang_disposeTranslationUnit(TU);
2458aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  clang_disposeIndex(CIdx);
2459aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  free(Locations);
2460aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  free_remapped_files(unsaved_files, num_unsaved_files);
2461aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  return 0;
2462aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis}
2463aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2464ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidisstatic enum CXVisitorResult findFileIncludesVisit(void *context,
2465ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis                                         CXCursor cursor, CXSourceRange range) {
2466ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  PrintCursor(cursor, NULL);
2467ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  PrintRange(range, "");
2468ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  printf("\n");
2469ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  return CXVisit_Continue;
2470ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis}
2471ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2472ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidisstatic int find_file_includes_in(int argc, const char **argv) {
2473ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  CXIndex CIdx;
2474ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  struct CXUnsavedFile *unsaved_files = 0;
2475ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  int num_unsaved_files = 0;
2476651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
2477ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  CXTranslationUnit TU;
2478ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  const char **Filenames = 0;
2479ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  unsigned NumFilenames = 0;
2480ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  unsigned Repeats = 1;
2481ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  unsigned I, FI;
2482ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2483ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  /* Count the number of locations. */
2484ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  while (strstr(argv[NumFilenames+1], "-file-includes-in=") == argv[NumFilenames+1])
2485ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    ++NumFilenames;
2486ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2487ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  /* Parse the locations. */
2488ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  assert(NumFilenames > 0 && "Unable to count filenames?");
2489ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  Filenames = (const char **)malloc(NumFilenames * sizeof(const char *));
2490ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  for (I = 0; I < NumFilenames; ++I) {
2491ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    const char *input = argv[I + 1] + strlen("-file-includes-in=");
2492ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    /* Copy the file name. */
2493ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    Filenames[I] = input;
2494ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  }
2495ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2496ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  if (parse_remapped_files(argc, argv, NumFilenames + 1, &unsaved_files,
2497ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis                           &num_unsaved_files))
2498ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    return -1;
2499ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2500ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING"))
2501ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    Repeats = 2;
2502ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2503ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  /* Parse the translation unit. When we're testing clang_getCursor() after
2504ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis     reparsing, don't remap unsaved files until the second parse. */
2505ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  CIdx = clang_createIndex(1, 1);
2506651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(
2507651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      CIdx, argv[argc - 1],
2508651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      argv + num_unsaved_files + 1 + NumFilenames,
2509651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      argc - num_unsaved_files - 2 - NumFilenames,
2510651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      unsaved_files,
2511651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Repeats > 1 ? 0 : num_unsaved_files, getDefaultParsingOptions(), &TU);
2512651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
2513651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
2514ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    fprintf(stderr, "unable to parse input\n");
2515651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
2516651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_disposeTranslationUnit(TU);
2517ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    return -1;
2518ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  }
2519ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2520ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  if (checkForErrors(TU) != 0)
2521ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    return -1;
2522ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2523ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  for (I = 0; I != Repeats; ++I) {
2524651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (Repeats > 1) {
2525651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Err = clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2526651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         clang_defaultReparseOptions(TU));
2527651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (Err != CXError_Success) {
2528651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        describeLibclangFailure(Err);
2529651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        clang_disposeTranslationUnit(TU);
2530651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        return 1;
2531651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      }
2532ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    }
2533ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2534ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    if (checkForErrors(TU) != 0)
2535ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      return -1;
2536ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2537ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    for (FI = 0; FI < NumFilenames; ++FI) {
2538ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      CXFile file = clang_getFile(TU, Filenames[FI]);
2539ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      if (!file)
2540ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        continue;
2541ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2542ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      if (checkForErrors(TU) != 0)
2543ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        return -1;
2544ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2545ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      if (I + 1 == Repeats) {
2546ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        CXCursorAndRangeVisitor visitor = { 0, findFileIncludesVisit };
2547ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        clang_findIncludesInFile(TU, file, visitor);
2548ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2549ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        if (checkForErrors(TU) != 0)
2550ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis          return -1;
2551ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      }
2552ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    }
2553ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  }
2554ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2555ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  PrintDiagnostics(TU);
2556ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  clang_disposeTranslationUnit(TU);
2557ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  clang_disposeIndex(CIdx);
25585256c1fd315104f7b1542bb231f16dbad8a8d64bArgyrios Kyrtzidis  free((void *)Filenames);
2559ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  free_remapped_files(unsaved_files, num_unsaved_files);
2560ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  return 0;
2561ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis}
2562ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
256311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis#define MAX_IMPORTED_ASTFILES 200
256411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
256511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidistypedef struct {
256611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  char **filenames;
256711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  unsigned num_files;
256811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis} ImportedASTFilesData;
256911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
257011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic ImportedASTFilesData *importedASTs_create() {
257111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  ImportedASTFilesData *p;
257211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p = malloc(sizeof(ImportedASTFilesData));
257311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p->filenames = malloc(MAX_IMPORTED_ASTFILES * sizeof(const char *));
257411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p->num_files = 0;
257511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  return p;
257611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis}
257711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
257811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic void importedASTs_dispose(ImportedASTFilesData *p) {
257911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  unsigned i;
258011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (!p)
258111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    return;
258211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
258311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  for (i = 0; i < p->num_files; ++i)
258411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    free(p->filenames[i]);
258511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  free(p->filenames);
258611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  free(p);
258711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis}
258811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
258911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic void importedASTS_insert(ImportedASTFilesData *p, const char *file) {
259011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  unsigned i;
259111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  assert(p && file);
259211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  for (i = 0; i < p->num_files; ++i)
259311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    if (strcmp(file, p->filenames[i]) == 0)
259411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis      return;
259511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  assert(p->num_files + 1 < MAX_IMPORTED_ASTFILES);
259611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p->filenames[p->num_files++] = strdup(file);
259711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis}
259811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
25996bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinestypedef struct IndexDataStringList_ {
26006bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  struct IndexDataStringList_ *next;
26016bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  char data[1]; /* Dynamically sized. */
26026bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines} IndexDataStringList;
26036bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
26044e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidistypedef struct {
26054e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *check_prefix;
26064e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  int first_check_printed;
2607dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  int fail_for_error;
26086f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  int abort;
260913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  const char *main_filename;
261011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  ImportedASTFilesData *importedASTs;
26116bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  IndexDataStringList *strings;
26126bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  CXTranslationUnit TU;
26134e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis} IndexData;
26144e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
26156bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic void free_client_data(IndexData *index_data) {
26166bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  IndexDataStringList *node = index_data->strings;
26176bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  while (node) {
26186bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    IndexDataStringList *next = node->next;
26196bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    free(node);
26206bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    node = next;
26216bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  }
26226bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data->strings = NULL;
26236bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines}
26246bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
26254e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void printCheck(IndexData *data) {
26264e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (data->check_prefix) {
26274e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    if (data->first_check_printed) {
26284e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      printf("// %s-NEXT: ", data->check_prefix);
26294e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    } else {
26304e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      printf("// %s     : ", data->check_prefix);
26314e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      data->first_check_printed = 1;
26324e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    }
26334e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
26344e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
26354e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2636dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic void printCXIndexFile(CXIdxClientFile file) {
263710f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXString filename = clang_getFileName((CXFile)file);
26384e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("%s", clang_getCString(filename));
26394e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_disposeString(filename);
26404e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
26414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
264213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidisstatic void printCXIndexLoc(CXIdxLoc loc, CXClientData client_data) {
264313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  IndexData *index_data;
26444e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString filename;
264513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  const char *cname;
2646dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  CXIdxClientFile file;
26474e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  unsigned line, column;
264813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  int isMainFile;
26494e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
265013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
26514e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_indexLoc_getFileLocation(loc, &file, 0, &line, &column, 0);
26524e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (line == 0) {
26538003fd613724847bba834ae41aca6f446af1f818Argyrios Kyrtzidis    printf("<invalid>");
26544e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    return;
26554e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
2656c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  if (!file) {
2657c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis    printf("<no idxfile>");
2658c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis    return;
2659c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  }
26604e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  filename = clang_getFileName((CXFile)file);
26614e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  cname = clang_getCString(filename);
266213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  if (strcmp(cname, index_data->main_filename) == 0)
266313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    isMainFile = 1;
266413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  else
266513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    isMainFile = 0;
266613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  clang_disposeString(filename);
266713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
266813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  if (!isMainFile) {
26694e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printCXIndexFile(file);
26704e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printf(":");
26714e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
26724e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("%d:%d", line, column);
26734e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
26744e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
267513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidisstatic unsigned digitCount(unsigned val) {
267613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  unsigned c = 1;
267713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  while (1) {
267813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    if (val < 10)
267913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis      return c;
268013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    ++c;
268113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    val /= 10;
268213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  }
268313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis}
268413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
26856bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic CXIdxClientContainer makeClientContainer(CXClientData *client_data,
26866bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                                const CXIdxEntityInfo *info,
26876ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                                CXIdxLoc loc) {
26886bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  IndexData *index_data;
26896bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  IndexDataStringList *node;
26904e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *name;
26914e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  char *newStr;
2692dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  CXIdxClientFile file;
26934e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  unsigned line, column;
26944e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2695dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  name = info->name;
26964e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (!name)
26974e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    name = "<anon-tag>";
26984e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
26994e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_indexLoc_getFileLocation(loc, &file, 0, &line, &column, 0);
27006bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
27016bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  node =
27026bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      (IndexDataStringList *)malloc(sizeof(IndexDataStringList) + strlen(name) +
27036bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                    digitCount(line) + digitCount(column) + 2);
27046bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  newStr = node->data;
27054e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  sprintf(newStr, "%s:%d:%d", name, line, column);
27066bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
27076bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  /* Remember string so it can be freed later. */
27086bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data = (IndexData *)client_data;
27096bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  node->next = index_data->strings;
27106bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data->strings = node;
27116bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
27126ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  return (CXIdxClientContainer)newStr;
27134e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
27144e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
27152957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidisstatic void printCXIndexContainer(const CXIdxContainerInfo *info) {
27162957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIdxClientContainer container;
27172957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  container = clang_index_getClientContainer(info);
27183e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis  if (!container)
27193e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis    printf("[<<NULL>>]");
27203e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis  else
27213e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis    printf("[%s]", (const char *)container);
27224e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
27234e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2724dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic const char *getEntityKindString(CXIdxEntityKind kind) {
2725dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  switch (kind) {
2726dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Unexposed: return "<<UNEXPOSED>>";
2727dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Typedef: return "typedef";
2728dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Function: return "function";
2729dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Variable: return "variable";
2730dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Field: return "field";
2731dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_EnumConstant: return "enumerator";
2732dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCClass: return "objc-class";
2733dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCProtocol: return "objc-protocol";
2734dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCCategory: return "objc-category";
2735c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  case CXIdxEntity_ObjCInstanceMethod: return "objc-instance-method";
2736c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  case CXIdxEntity_ObjCClassMethod: return "objc-class-method";
2737dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCProperty: return "objc-property";
2738dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCIvar: return "objc-ivar";
2739dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Enum: return "enum";
2740dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Struct: return "struct";
2741dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Union: return "union";
2742dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_CXXClass: return "c++-class";
27432957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXNamespace: return "namespace";
27442957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXNamespaceAlias: return "namespace-alias";
27452957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXStaticVariable: return "c++-static-var";
27462957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXStaticMethod: return "c++-static-method";
27472957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXInstanceMethod: return "c++-instance-method";
27482957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXConstructor: return "constructor";
27492957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXDestructor: return "destructor";
27502957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXConversionFunction: return "conversion-func";
27512957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXTypeAlias: return "type-alias";
275235adca009c65a9578b91ab0bdca13691624e13afDavid Blaikie  case CXIdxEntity_CXXInterface: return "c++-__interface";
27532957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  }
27542957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  assert(0 && "Garbage entity kind");
27552957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  return 0;
27562957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis}
27572957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis
27582957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidisstatic const char *getEntityTemplateKindString(CXIdxEntityCXXTemplateKind kind) {
27592957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  switch (kind) {
27602957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_NonTemplate: return "";
27612957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_Template: return "-template";
27622957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_TemplatePartialSpecialization:
27632957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    return "-template-partial-spec";
27642957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_TemplateSpecialization: return "-template-spec";
2765dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
27666ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  assert(0 && "Garbage entity kind");
27676ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  return 0;
2768dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis}
2769dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2770838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidisstatic const char *getEntityLanguageString(CXIdxEntityLanguage kind) {
2771838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  switch (kind) {
2772838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_None: return "<none>";
2773838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_C: return "C";
2774838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_ObjC: return "ObjC";
2775838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_CXX: return "C++";
2776838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  }
2777838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  assert(0 && "Garbage language kind");
2778838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  return 0;
2779838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis}
2780838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis
2781dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic void printEntityInfo(const char *cb,
2782dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                            CXClientData client_data,
27836ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                            const CXIdxEntityInfo *info) {
27844e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *name;
27854e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
2786643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  unsigned i;
27874e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
27884e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
27894e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2790c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis  if (!info) {
2791c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis    printf("%s: <<NULL>>", cb);
2792c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis    return;
2793c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis  }
2794c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis
2795dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  name = info->name;
27964e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (!name)
27974e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    name = "<anon-tag>";
27984e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
27992957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  printf("%s: kind: %s%s", cb, getEntityKindString(info->kind),
28002957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis         getEntityTemplateKindString(info->templateKind));
2801dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | name: %s", name);
2802dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | USR: %s", info->USR);
2803c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  printf(" | lang: %s", getEntityLanguageString(info->lang));
2804643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis
2805643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  for (i = 0; i != info->numAttributes; ++i) {
2806643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis    const CXIdxAttrInfo *Attr = info->attributes[i];
2807643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis    printf("     <attribute>: ");
2808f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(Attr->cursor, NULL);
2809643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  }
28104e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
28114e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2812b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidisstatic void printBaseClassInfo(CXClientData client_data,
2813b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis                               const CXIdxBaseClassInfo *info) {
2814b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printEntityInfo("     <base>", client_data, info->base);
2815b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printf(" | cursor: ");
2816f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
2817b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printf(" | loc: ");
281813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2819b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis}
2820b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis
2821c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidisstatic void printProtocolList(const CXIdxObjCProtocolRefListInfo *ProtoInfo,
2822c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis                              CXClientData client_data) {
2823c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  unsigned i;
2824c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  for (i = 0; i < ProtoInfo->numProtocols; ++i) {
2825c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printEntityInfo("     <protocol>", client_data,
2826c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis                    ProtoInfo->protocols[i]->protocol);
2827c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf(" | cursor: ");
2828f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(ProtoInfo->protocols[i]->cursor, NULL);
2829c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf(" | loc: ");
283013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    printCXIndexLoc(ProtoInfo->protocols[i]->loc, client_data);
2831c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf("\n");
2832c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  }
2833c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis}
2834c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis
28354e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void index_diagnostic(CXClientData client_data,
2836996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis                             CXDiagnosticSet diagSet, void *reserved) {
28374e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString str;
28384e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *cstr;
2839996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  unsigned numDiags, i;
2840996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  CXDiagnostic diag;
28414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
28424e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
28434e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
28444e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2845996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  numDiags = clang_getNumDiagnosticsInSet(diagSet);
2846996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  for (i = 0; i != numDiags; ++i) {
2847996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    diag = clang_getDiagnosticInSet(diagSet, i);
2848996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    str = clang_formatDiagnostic(diag, clang_defaultDiagnosticDisplayOptions());
2849996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    cstr = clang_getCString(str);
2850996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    printf("[diagnostic]: %s\n", cstr);
2851996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    clang_disposeString(str);
2852996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis
2853996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    if (getenv("CINDEXTEST_FAILONERROR") &&
2854996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis        clang_getDiagnosticSeverity(diag) >= CXDiagnostic_Error) {
2855996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis      index_data->fail_for_error = 1;
2856996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    }
2857dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
28584e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
28594e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2860dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientFile index_enteredMainFile(CXClientData client_data,
2861dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                                       CXFile file, void *reserved) {
2862dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  IndexData *index_data;
286362d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis  CXString filename;
286462d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis
2865dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
2866dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCheck(index_data);
2867dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
286862d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis  filename = clang_getFileName(file);
286913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  index_data->main_filename = clang_getCString(filename);
287013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  clang_disposeString(filename);
287113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
2872dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf("[enteredMainFile]: ");
2873dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)file);
2874dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf("\n");
2875dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2876dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientFile)file;
28774e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
28784e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2879dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientFile index_ppIncludedFile(CXClientData client_data,
28806ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                            const CXIdxIncludedFileInfo *info) {
28814e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
28826bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  CXModule Mod;
28834e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
28844e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
28854e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
288666042b32b3b37ddcba731ff05c2792e3bb572102Argyrios Kyrtzidis  printf("[ppIncludedFile]: ");
2887dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)info->file);
28884e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | name: \"%s\"", info->filename);
28894e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | hash loc: ");
289013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->hashLoc, client_data);
28916bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  printf(" | isImport: %d | isAngled: %d | isModule: %d",
28928d7a24e94b58676e57fd3f47353cbdbc59917d81Argyrios Kyrtzidis         info->isImport, info->isAngled, info->isModuleImport);
28936bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
28946bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  Mod = clang_getModuleForFile(index_data->TU, (CXFile)info->file);
28956bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  if (Mod) {
28966bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    CXString str = clang_Module_getFullName(Mod);
28976bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    const char *cstr = clang_getCString(str);
28986bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    printf(" | module: %s", cstr);
28996bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    clang_disposeString(str);
29006bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  }
29016bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
29026bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  printf("\n");
2903dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2904dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientFile)info->file;
29054e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
29064e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
29072c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidisstatic CXIdxClientFile index_importedASTFile(CXClientData client_data,
29082c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis                                         const CXIdxImportedASTFileInfo *info) {
29092c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  IndexData *index_data;
29102c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
29112c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  printCheck(index_data);
29122c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis
291311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (index_data->importedASTs) {
291410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString filename = clang_getFileName(info->file);
291511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    importedASTS_insert(index_data->importedASTs, clang_getCString(filename));
291611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    clang_disposeString(filename);
291711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  }
291811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
29192c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  printf("[importedASTFile]: ");
29202c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)info->file);
2921134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis  if (info->module) {
292210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString name = clang_Module_getFullName(info->module);
2923134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printf(" | loc: ");
2924134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printCXIndexLoc(info->loc, client_data);
2925134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printf(" | name: \"%s\"", clang_getCString(name));
2926134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printf(" | isImplicit: %d\n", info->isImplicit);
2927134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    clang_disposeString(name);
2928900ab95e12bb7483971640a91075699eec391804Argyrios Kyrtzidis  } else {
29293c5527e939c454bfd3e16354908e0c4315be5d2fNAKAMURA Takumi    /* PCH file, the rest are not relevant. */
2930900ab95e12bb7483971640a91075699eec391804Argyrios Kyrtzidis    printf("\n");
2931134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis  }
29322c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis
29332c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  return (CXIdxClientFile)info->file;
29342c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis}
29352c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis
29366bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic CXIdxClientContainer
29376bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesindex_startedTranslationUnit(CXClientData client_data, void *reserved) {
29384e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
29394e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
29404e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
29414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
294266042b32b3b37ddcba731ff05c2792e3bb572102Argyrios Kyrtzidis  printf("[startedTranslationUnit]\n");
2943dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientContainer)"TU";
29444e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
29454e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
29466ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidisstatic void index_indexDeclaration(CXClientData client_data,
29472957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                                   const CXIdxDeclInfo *info) {
2948dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  IndexData *index_data;
29496ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  const CXIdxObjCCategoryDeclInfo *CatInfo;
29506ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  const CXIdxObjCInterfaceDeclInfo *InterInfo;
2951c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  const CXIdxObjCProtocolRefListInfo *ProtoInfo;
2952792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  const CXIdxObjCPropertyDeclInfo *PropInfo;
2953b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  const CXIdxCXXClassDeclInfo *CXXClassInfo;
2954b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  unsigned i;
2955dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
29564e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2957dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo("[indexDeclaration]", client_data, info->entityInfo);
2958dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | cursor: ");
2959f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
2960dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | loc: ");
296113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2962b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printf(" | semantic-container: ");
2963b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printCXIndexContainer(info->semanticContainer);
2964b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printf(" | lexical-container: ");
2965b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printCXIndexContainer(info->lexicalContainer);
2966dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | isRedecl: %d", info->isRedeclaration);
2967c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isDef: %d", info->isDefinition);
2968838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  if (info->flags & CXIdxDeclFlag_Skipped) {
2969838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    assert(!info->isContainer);
2970838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    printf(" | isContainer: skipped");
2971838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  } else {
2972838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    printf(" | isContainer: %d", info->isContainer);
2973838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  }
2974c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isImplicit: %d\n", info->isImplicit);
29754e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2976b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  for (i = 0; i != info->numAttributes; ++i) {
297787adb0bf9375390de4c66d9e2ad110cc492cd655NAKAMURA Takumi    const CXIdxAttrInfo *Attr = info->attributes[i];
2978b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis    printf("     <attribute>: ");
2979f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(Attr->cursor, NULL);
2980b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis    printf("\n");
2981b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  }
2982b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis
2983dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  if (clang_index_isEntityObjCContainerKind(info->entityInfo->kind)) {
2984dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    const char *kindName = 0;
2985dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    CXIdxObjCContainerKind K = clang_index_getObjCContainerDeclInfo(info)->kind;
2986dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    switch (K) {
2987dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_ForwardRef:
2988dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "forward-ref"; break;
2989dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_Interface:
2990dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "interface"; break;
2991dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_Implementation:
2992dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "implementation"; break;
2993dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    }
2994dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printCheck(index_data);
2995dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printf("     <ObjCContainerInfo>: kind: %s\n", kindName);
2996dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
29974e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
29986ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  if ((CatInfo = clang_index_getObjCCategoryDeclInfo(info))) {
2999dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printEntityInfo("     <ObjCCategoryInfo>: class", client_data,
3000dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                    CatInfo->objcClass);
300121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    printf(" | cursor: ");
3002f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(CatInfo->classCursor, NULL);
300321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    printf(" | loc: ");
300413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    printCXIndexLoc(CatInfo->classLoc, client_data);
3005dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printf("\n");
3006dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
30074e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
30086ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  if ((InterInfo = clang_index_getObjCInterfaceDeclInfo(info))) {
30096ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis    if (InterInfo->superInfo) {
3010b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printBaseClassInfo(client_data, InterInfo->superInfo);
30116ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis      printf("\n");
30126ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis    }
30134e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
30144e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3015c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  if ((ProtoInfo = clang_index_getObjCProtocolRefListInfo(info))) {
3016c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printProtocolList(ProtoInfo, client_data);
30176ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  }
30184e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3019792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  if ((PropInfo = clang_index_getObjCPropertyDeclInfo(info))) {
3020792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    if (PropInfo->getter) {
3021792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printEntityInfo("     <getter>", client_data, PropInfo->getter);
3022792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printf("\n");
3023792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    }
3024792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    if (PropInfo->setter) {
3025792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printEntityInfo("     <setter>", client_data, PropInfo->setter);
3026792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printf("\n");
3027792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    }
3028792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  }
3029792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis
3030b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  if ((CXXClassInfo = clang_index_getCXXClassDeclInfo(info))) {
3031b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis    for (i = 0; i != CXXClassInfo->numBases; ++i) {
3032b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printBaseClassInfo(client_data, CXXClassInfo->bases[i]);
3033b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printf("\n");
3034b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis    }
3035b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  }
3036b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis
30372957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  if (info->declAsContainer)
30386bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    clang_index_setClientContainer(
30396bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        info->declAsContainer,
30406bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        makeClientContainer(client_data, info->entityInfo, info->loc));
30414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
30424e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
30434e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void index_indexEntityReference(CXClientData client_data,
30446ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                       const CXIdxEntityRefInfo *info) {
30456bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  printEntityInfo("[indexEntityReference]", client_data,
30466bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                  info->referencedEntity);
30474e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | cursor: ");
3048f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
30494e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | loc: ");
305013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
3051dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo(" | <parent>:", client_data, info->parentEntity);
30524e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | container: ");
30534e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCXIndexContainer(info->container);
3054c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | refkind: ");
3055aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  switch (info->kind) {
3056aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  case CXIdxEntityRef_Direct: printf("direct"); break;
3057b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  case CXIdxEntityRef_Implicit: printf("implicit"); break;
3058aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  }
30594e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("\n");
30604e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
30614e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
30626f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidisstatic int index_abortQuery(CXClientData client_data, void *reserved) {
30636f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  IndexData *index_data;
30646f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
30656f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  return index_data->abort;
30666f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis}
30676f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis
30684e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic IndexerCallbacks IndexCB = {
30696f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_abortQuery,
30704e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_diagnostic,
3071dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_enteredMainFile,
30724e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_ppIncludedFile,
30732c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  index_importedASTFile,
30744e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_startedTranslationUnit,
3075dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_indexDeclaration,
30764e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_indexEntityReference
30774e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis};
30784e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
307922490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidisstatic unsigned getIndexOptions(void) {
308022490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  unsigned index_opts;
308122490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  index_opts = 0;
308222490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  if (getenv("CINDEXTEST_SUPPRESSREFS"))
308322490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis    index_opts |= CXIndexOpt_SuppressRedundantRefs;
308422490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  if (getenv("CINDEXTEST_INDEXLOCALSYMBOLS"))
308522490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis    index_opts |= CXIndexOpt_IndexFunctionLocalSymbols;
3086838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  if (!getenv("CINDEXTEST_DISABLE_SKIPPARSEDBODIES"))
3087838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    index_opts |= CXIndexOpt_SkipParsedBodiesInSession;
308822490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis
308922490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  return index_opts;
309022490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis}
309122490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis
3092d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidisstatic int index_compile_args(int num_args, const char **args,
3093d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              CXIndexAction idxAction,
3094d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              ImportedASTFilesData *importedASTs,
3095d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              const char *check_prefix) {
3096d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  IndexData index_data;
3097d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  unsigned index_opts;
3098d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  int result;
3099d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3100d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (num_args == 0) {
3101d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    fprintf(stderr, "no compiler arguments\n");
3102d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return -1;
3103d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
3104d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3105d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.check_prefix = check_prefix;
3106d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.first_check_printed = 0;
3107d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.fail_for_error = 0;
3108d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.abort = 0;
3109d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.main_filename = "";
3110d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.importedASTs = importedASTs;
31116bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data.strings = NULL;
31126bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data.TU = NULL;
3113d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3114d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_opts = getIndexOptions();
3115d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = clang_indexSourceFile(idxAction, &index_data,
3116d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                 &IndexCB,sizeof(IndexCB), index_opts,
3117d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                 0, args, num_args, 0, 0, 0,
3118d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                 getDefaultParsingOptions());
3119651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (result != CXError_Success)
3120651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(result);
3121651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
3122d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (index_data.fail_for_error)
3123d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    result = -1;
3124d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
31256bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  free_client_data(&index_data);
3126d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return result;
3127d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis}
3128d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3129d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidisstatic int index_ast_file(const char *ast_file,
3130d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          CXIndex Idx,
3131d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          CXIndexAction idxAction,
3132d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          ImportedASTFilesData *importedASTs,
3133d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          const char *check_prefix) {
3134d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  CXTranslationUnit TU;
3135d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  IndexData index_data;
3136d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  unsigned index_opts;
3137d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  int result;
3138d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3139d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (!CreateTranslationUnit(Idx, ast_file, &TU))
3140d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return -1;
3141d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3142d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.check_prefix = check_prefix;
3143d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.first_check_printed = 0;
3144d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.fail_for_error = 0;
3145d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.abort = 0;
3146d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.main_filename = "";
3147d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.importedASTs = importedASTs;
31486bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data.strings = NULL;
31496bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data.TU = TU;
3150d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3151d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_opts = getIndexOptions();
3152d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = clang_indexTranslationUnit(idxAction, &index_data,
3153d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                      &IndexCB,sizeof(IndexCB),
3154d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                      index_opts, TU);
3155d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (index_data.fail_for_error)
3156d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    result = -1;
3157d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3158d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  clang_disposeTranslationUnit(TU);
31596bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  free_client_data(&index_data);
3160d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return result;
3161d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis}
3162d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
316311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic int index_file(int argc, const char **argv, int full) {
31644e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *check_prefix;
31652957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndex Idx;
31662957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndexAction idxAction;
3167d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  ImportedASTFilesData *importedASTs;
3168dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  int result;
31694e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
31704e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  check_prefix = 0;
31714e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc > 0) {
31724e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
31734e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
31744e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      ++argv;
31754e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      --argc;
31764e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    }
31774e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
31784e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
31792957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
3180fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                                /* displayDiagnostics=*/1))) {
31812957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
31822957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    return 1;
31832957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  }
3184d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
3185d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  importedASTs = 0;
318611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (full)
3187d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    importedASTs = importedASTs_create();
3188d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3189d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = index_compile_args(argc, argv, idxAction, importedASTs, check_prefix);
3190d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (result != 0)
3191d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    goto finished;
3192dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
319311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (full) {
319411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    unsigned i;
3195d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    for (i = 0; i < importedASTs->num_files && result == 0; ++i) {
3196d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      result = index_ast_file(importedASTs->filenames[i], Idx, idxAction,
3197d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              importedASTs, check_prefix);
319811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    }
319911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  }
320021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
320111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisfinished:
3202d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  importedASTs_dispose(importedASTs);
32032957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
32042957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_disposeIndex(Idx);
320521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  return result;
320621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis}
320721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
320821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidisstatic int index_tu(int argc, const char **argv) {
3209d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  const char *check_prefix;
321021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  CXIndex Idx;
32112957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndexAction idxAction;
321221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  int result;
321321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
321421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  check_prefix = 0;
321521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc > 0) {
321621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
321721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
321821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      ++argv;
321921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      --argc;
322021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    }
322121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
322221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
3223d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
3224fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                                /* displayDiagnostics=*/1))) {
3225d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
3226d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return 1;
3227d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
3228d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
3229d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3230d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = index_ast_file(argv[0], Idx, idxAction,
3231d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          /*importedASTs=*/0, check_prefix);
3232d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3233d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
3234d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  clang_disposeIndex(Idx);
3235d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return result;
3236d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis}
3237d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3238d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidisstatic int index_compile_db(int argc, const char **argv) {
3239d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  const char *check_prefix;
3240d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  CXIndex Idx;
3241d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  CXIndexAction idxAction;
3242d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  int errorCode = 0;
3243d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3244d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  check_prefix = 0;
3245d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (argc > 0) {
3246d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
3247d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
3248d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      ++argv;
3249d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      --argc;
3250d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    }
3251d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
3252d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
325321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc == 0) {
3254d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    fprintf(stderr, "no compilation database\n");
325521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return -1;
325621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
325721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
325821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
3259fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                                /* displayDiagnostics=*/1))) {
326021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
326121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return 1;
326221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
3263d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
326421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
3265d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  {
3266d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    const char *database = argv[0];
3267d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompilationDatabase db = 0;
3268d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompileCommands CCmds = 0;
3269d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompileCommand CCmd;
3270d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompilationDatabase_Error ec;
3271d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXString wd;
3272d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#define MAX_COMPILE_ARGS 512
3273d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXString cxargs[MAX_COMPILE_ARGS];
3274d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    const char *args[MAX_COMPILE_ARGS];
3275d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    char *tmp;
3276d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    unsigned len;
3277d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    char *buildDir;
3278d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    int i, a, numCmds, numArgs;
3279d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3280d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    len = strlen(database);
3281d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    tmp = (char *) malloc(len+1);
3282d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    memcpy(tmp, database, len+1);
3283d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    buildDir = dirname(tmp);
3284d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3285d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    db = clang_CompilationDatabase_fromDirectory(buildDir, &ec);
3286d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3287d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    if (db) {
3288d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3289d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      if (ec!=CXCompilationDatabase_NoError) {
3290d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        printf("unexpected error %d code while loading compilation database\n", ec);
3291d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = -1;
3292d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        goto cdb_end;
3293d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
329421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
32952bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis      if (chdir(buildDir) != 0) {
32962bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        printf("Could not chdir to %s\n", buildDir);
32972bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        errorCode = -1;
32982bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        goto cdb_end;
32992bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis      }
330021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
33012bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis      CCmds = clang_CompilationDatabase_getAllCompileCommands(db);
3302d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      if (!CCmds) {
3303d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        printf("compilation db is empty\n");
3304d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = -1;
3305d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        goto cdb_end;
3306d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
3307d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3308d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      numCmds = clang_CompileCommands_getSize(CCmds);
3309d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3310d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      if (numCmds==0) {
3311d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        fprintf(stderr, "should not get an empty compileCommand set\n");
3312d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = -1;
3313d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        goto cdb_end;
3314d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
3315d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3316d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      for (i=0; i<numCmds && errorCode == 0; ++i) {
3317d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        CCmd = clang_CompileCommands_getCommand(CCmds, i);
3318d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3319d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        wd = clang_CompileCommand_getDirectory(CCmd);
33202bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        if (chdir(clang_getCString(wd)) != 0) {
33212bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis          printf("Could not chdir to %s\n", clang_getCString(wd));
33222bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis          errorCode = -1;
33232bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis          goto cdb_end;
33242bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        }
3325d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        clang_disposeString(wd);
3326d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3327d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        numArgs = clang_CompileCommand_getNumArgs(CCmd);
3328d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        if (numArgs > MAX_COMPILE_ARGS){
3329d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          fprintf(stderr, "got more compile arguments than maximum\n");
3330d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          errorCode = -1;
3331d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          goto cdb_end;
3332d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        }
3333d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        for (a=0; a<numArgs; ++a) {
3334d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          cxargs[a] = clang_CompileCommand_getArg(CCmd, a);
3335d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          args[a] = clang_getCString(cxargs[a]);
3336d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        }
3337d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3338d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = index_compile_args(numArgs, args, idxAction,
3339d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                       /*importedASTs=*/0, check_prefix);
3340d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3341d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        for (a=0; a<numArgs; ++a)
3342d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          clang_disposeString(cxargs[a]);
3343d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
3344d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    } else {
3345d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      printf("database loading failed with error code %d.\n", ec);
3346d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      errorCode = -1;
3347d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    }
3348d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3349d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  cdb_end:
3350d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    clang_CompileCommands_dispose(CCmds);
3351d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    clang_CompilationDatabase_dispose(db);
3352d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    free(tmp);
3353d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3354d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
33554e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
33562957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
33572957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_disposeIndex(Idx);
3358d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return errorCode;
33594e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
33604e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3361fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregorint perform_token_annotation(int argc, const char **argv) {
3362fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  const char *input = argv[1];
3363fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  char *filename = 0;
3364fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned line, second_line;
3365fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned column, second_column;
3366fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXIndex CIdx;
3367fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXTranslationUnit TU = 0;
3368fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  int errorCode;
3369fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
3370fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  int num_unsaved_files = 0;
3371fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXToken *tokens;
3372fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned num_tokens;
3373fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXSourceRange range;
3374fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXSourceLocation startLoc, endLoc;
3375fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXFile file = 0;
3376fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXCursor *cursors = 0;
3377651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  CXSourceRangeList *skipped_ranges = 0;
3378651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
3379fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned i;
3380fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3381fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  input += strlen("-test-annotate-tokens=");
3382fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if ((errorCode = parse_file_line_column(input, &filename, &line, &column,
3383fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                          &second_line, &second_column)))
3384fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return errorCode;
3385fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3386e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  if (parse_remapped_files(argc, argv, 2, &unsaved_files, &num_unsaved_files)) {
3387e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    free(filename);
3388fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return -1;
3389e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  }
3390fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
33910a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  CIdx = clang_createIndex(0, 1);
3392651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(CIdx, argv[argc - 1],
3393651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + num_unsaved_files + 2,
3394651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - num_unsaved_files - 3,
3395651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    unsaved_files,
3396651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    num_unsaved_files,
3397651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    getDefaultParsingOptions(), &TU);
3398651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
3399fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    fprintf(stderr, "unable to parse input\n");
3400651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
3401fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    clang_disposeIndex(CIdx);
3402fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    free(filename);
3403fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
3404fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return -1;
3405e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
3406fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  errorCode = 0;
3407fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3408e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  if (checkForErrors(TU) != 0) {
3409e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    errorCode = -1;
3410e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    goto teardown;
3411e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  }
3412dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3413ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING")) {
3414ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    for (i = 0; i < 5; ++i) {
3415651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Err = clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
3416651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         clang_defaultReparseOptions(TU));
3417651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (Err != CXError_Success) {
3418ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        fprintf(stderr, "Unable to reparse translation unit!\n");
3419651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        describeLibclangFailure(Err);
3420ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        errorCode = -1;
3421ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        goto teardown;
3422ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      }
3423ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    }
3424ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
3425ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
3426dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
3427dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
3428dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
3429dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
3430dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3431fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  file = clang_getFile(TU, filename);
3432fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!file) {
3433fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    fprintf(stderr, "file %s is not in this translation unit\n", filename);
3434fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
3435fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    goto teardown;
3436fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
3437fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3438fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  startLoc = clang_getLocation(TU, file, line, column);
3439fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (clang_equalLocations(clang_getNullLocation(), startLoc)) {
3440e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(stderr, "invalid source location %s:%d:%d\n", filename, line,
3441fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            column);
3442fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
3443e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    goto teardown;
3444fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
3445fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3446fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  endLoc = clang_getLocation(TU, file, second_line, second_column);
3447fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (clang_equalLocations(clang_getNullLocation(), endLoc)) {
3448e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(stderr, "invalid source location %s:%d:%d\n", filename,
3449fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            second_line, second_column);
3450fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
3451e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    goto teardown;
3452fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
3453fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3454fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  range = clang_getRange(startLoc, endLoc);
3455fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_tokenize(TU, range, &tokens, &num_tokens);
3456dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3457dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
3458dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
3459dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
3460dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
3461dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3462fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  cursors = (CXCursor *)malloc(num_tokens * sizeof(CXCursor));
3463fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_annotateTokens(TU, tokens, num_tokens, cursors);
3464dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3465dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
3466dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
3467dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
3468dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
3469dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3470651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  skipped_ranges = clang_getSkippedRanges(TU, file);
3471651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  for (i = 0; i != skipped_ranges->count; ++i) {
3472651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    unsigned start_line, start_column, end_line, end_column;
3473651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_getSpellingLocation(clang_getRangeStart(skipped_ranges->ranges[i]),
3474651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                              0, &start_line, &start_column, 0);
3475651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_getSpellingLocation(clang_getRangeEnd(skipped_ranges->ranges[i]),
3476651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                              0, &end_line, &end_column, 0);
3477651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    printf("Skipping: ");
3478651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    PrintExtent(stdout, start_line, start_column, end_line, end_column);
3479651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    printf("\n");
3480651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  }
3481651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  clang_disposeSourceRangeList(skipped_ranges);
3482651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
3483fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  for (i = 0; i != num_tokens; ++i) {
3484fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *kind = "<unknown>";
348510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString spelling = clang_getTokenSpelling(TU, tokens[i]);
348610f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceRange extent = clang_getTokenExtent(TU, tokens[i]);
3487fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    unsigned start_line, start_column, end_line, end_column;
3488fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3489fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    switch (clang_getTokenKind(tokens[i])) {
3490fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Punctuation: kind = "Punctuation"; break;
3491fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Keyword: kind = "Keyword"; break;
3492fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Identifier: kind = "Identifier"; break;
3493fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Literal: kind = "Literal"; break;
3494fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Comment: kind = "Comment"; break;
3495fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
3496a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(clang_getRangeStart(extent),
3497a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              0, &start_line, &start_column, 0);
3498a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(clang_getRangeEnd(extent),
3499a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              0, &end_line, &end_column, 0);
350051b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar    printf("%s: \"%s\" ", kind, clang_getCString(spelling));
3501342742adb69af8ef8f2c652f162496b98c45bc20Benjamin Kramer    clang_disposeString(spelling);
350251b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar    PrintExtent(stdout, start_line, start_column, end_line, end_column);
35030045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    if (!clang_isInvalid(cursors[i].kind)) {
35040045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor      printf(" ");
3505f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko      PrintCursor(cursors[i], NULL);
35060045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    }
35070045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    printf("\n");
3508fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
3509fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free(cursors);
351093f5e6a5d7690f90bc8a94e6b40d6c7d19719e0cTed Kremenek  clang_disposeTokens(TU, tokens, num_tokens);
3511fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3512fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor teardown:
3513a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
3514fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_disposeTranslationUnit(TU);
3515fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_disposeIndex(CIdx);
3516fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free(filename);
3517fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
3518fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  return errorCode;
3519fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
3520fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3521db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonstatic int
3522db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonperform_test_compilation_db(const char *database, int argc, const char **argv) {
3523db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompilationDatabase db;
3524db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompileCommands CCmds;
3525db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompileCommand CCmd;
3526db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompilationDatabase_Error ec;
3527db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXString wd;
3528db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXString arg;
3529db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  int errorCode = 0;
3530db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  char *tmp;
3531db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  unsigned len;
3532db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  char *buildDir;
3533db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  int i, j, a, numCmds, numArgs;
3534db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3535db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  len = strlen(database);
3536db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  tmp = (char *) malloc(len+1);
3537db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  memcpy(tmp, database, len+1);
3538db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  buildDir = dirname(tmp);
3539db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3540c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison  db = clang_CompilationDatabase_fromDirectory(buildDir, &ec);
3541db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3542db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  if (db) {
3543db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3544db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    if (ec!=CXCompilationDatabase_NoError) {
3545db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      printf("unexpected error %d code while loading compilation database\n", ec);
3546db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      errorCode = -1;
3547db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      goto cdb_end;
3548db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    }
3549db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3550db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    for (i=0; i<argc && errorCode==0; ) {
3551db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      if (strcmp(argv[i],"lookup")==0){
3552c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        CCmds = clang_CompilationDatabase_getCompileCommands(db, argv[i+1]);
3553db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3554db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (!CCmds) {
3555db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("file %s not found in compilation db\n", argv[i+1]);
3556db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          errorCode = -1;
3557db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          break;
3558db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
3559db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3560c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        numCmds = clang_CompileCommands_getSize(CCmds);
3561db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3562db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (numCmds==0) {
3563db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          fprintf(stderr, "should not get an empty compileCommand set for file"
3564db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison                          " '%s'\n", argv[i+1]);
3565db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          errorCode = -1;
3566db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          break;
3567db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
3568db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3569db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        for (j=0; j<numCmds; ++j) {
3570c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          CCmd = clang_CompileCommands_getCommand(CCmds, j);
3571db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3572c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          wd = clang_CompileCommand_getDirectory(CCmd);
3573db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("workdir:'%s'", clang_getCString(wd));
3574db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          clang_disposeString(wd);
3575db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3576db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf(" cmdline:'");
3577c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          numArgs = clang_CompileCommand_getNumArgs(CCmd);
3578db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          for (a=0; a<numArgs; ++a) {
3579db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            if (a) printf(" ");
3580c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison            arg = clang_CompileCommand_getArg(CCmd, a);
3581db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            printf("%s", clang_getCString(arg));
3582db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            clang_disposeString(arg);
3583db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          }
3584db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("'\n");
3585db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
3586db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3587c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        clang_CompileCommands_dispose(CCmds);
3588db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3589db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        i += 2;
3590db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      }
3591db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    }
3592c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison    clang_CompilationDatabase_dispose(db);
3593db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  } else {
3594db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    printf("database loading failed with error code %d.\n", ec);
3595db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    errorCode = -1;
3596db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  }
3597db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3598db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisoncdb_end:
3599db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  free(tmp);
3600db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3601db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  return errorCode;
3602db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison}
3603db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
36040d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
3605f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/* USR printing.                                                              */
3606f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/******************************************************************************/
3607f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3608f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic int insufficient_usr(const char *kind, const char *usage) {
3609f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "USR for '%s' requires: %s\n", kind, usage);
3610f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 1;
3611f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3612f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3613f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic unsigned isUSR(const char *s) {
3614f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return s[0] == 'c' && s[1] == ':';
3615f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3616f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3617f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic int not_usr(const char *s, const char *arg) {
3618f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "'%s' argument ('%s') is not a USR\n", s, arg);
3619f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 1;
3620f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3621f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3622f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic void print_usr(CXString usr) {
3623f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  const char *s = clang_getCString(usr);
3624f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  printf("%s\n", s);
3625f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  clang_disposeString(usr);
3626f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3627f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3628f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic void display_usrs() {
3629f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "-print-usrs options:\n"
3630f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCCategory <class name> <category name>\n"
3631f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCClass <class name>\n"
3632f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCIvar <ivar name> <class USR>\n"
3633f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCMethod <selector> [0=class method|1=instance method] "
3634f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            "<class USR>\n"
3635f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          " ObjCProperty <property name> <class USR>\n"
3636f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          " ObjCProtocol <protocol name>\n");
3637f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3638f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3639f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekint print_usrs(const char **I, const char **E) {
3640f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  while (I != E) {
3641f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    const char *kind = *I;
3642f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    unsigned len = strlen(kind);
3643f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    switch (len) {
3644f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 8:
3645f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCIvar", 8) == 0) {
3646f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
3647f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<ivar name> <class USR>");
3648f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[2]))
3649f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[2]);
3650f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
3651f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
3652a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[2];
3653ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
3654f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCIvar(I[1], x));
3655f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
3656f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3657f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
3658f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3659f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3660f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3661f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 9:
3662f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCClass", 9) == 0) {
3663f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 1 >= E)
3664f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<class name>");
3665f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCClass(I[1]));
3666f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 2;
3667f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3668f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3669f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3670f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 10:
3671f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCMethod", 10) == 0) {
3672f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 3 >= E)
3673f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<method selector> "
3674f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek                "[0=class method|1=instance method] <class USR>");
3675f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[3]))
3676f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[3]);
3677f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
3678f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
3679a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[3];
3680ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
3681f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCMethod(I[1], atoi(I[2]), x));
3682f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
3683f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 4;
3684f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3685f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3686f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3687f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 12:
3688f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCCategory", 12) == 0) {
3689f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
3690f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<class name> <category name>");
3691f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCCategory(I[1], I[2]));
3692f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
3693f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3694f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3695f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCProtocol", 12) == 0) {
3696f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 1 >= E)
3697f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<protocol name>");
3698f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCProtocol(I[1]));
3699f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 2;
3700f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3701f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3702f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCProperty", 12) == 0) {
3703f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
3704f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<property name> <class USR>");
3705f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[2]))
3706f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[2]);
3707f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
3708f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
3709a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[2];
3710ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
3711f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCProperty(I[1], x));
3712f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
3713f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
3714f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3715f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3716f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3717f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      default:
3718f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3719f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
3720f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    break;
3721f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3722f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3723f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  if (I != E) {
3724f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    fprintf(stderr, "Invalid USR kind: %s\n", *I);
3725f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    display_usrs();
3726f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return 1;
3727f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3728f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 0;
3729f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3730f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3731f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekint print_usrs_file(const char *file_name) {
3732f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  char line[2048];
3733f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  const char *args[128];
3734f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  unsigned numChars = 0;
3735f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3736f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  FILE *fp = fopen(file_name, "r");
3737f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  if (!fp) {
3738f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    fprintf(stderr, "error: cannot open '%s'\n", file_name);
3739f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return 1;
3740f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3741f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3742f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  /* This code is not really all that safe, but it works fine for testing. */
3743f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  while (!feof(fp)) {
3744f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    char c = fgetc(fp);
3745f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    if (c == '\n') {
3746f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      unsigned i = 0;
3747f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      const char *s = 0;
3748f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3749f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (numChars == 0)
3750f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        continue;
3751f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3752f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      line[numChars] = '\0';
3753f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      numChars = 0;
3754f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3755f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (line[0] == '/' && line[1] == '/')
3756f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        continue;
3757f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3758f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      s = strtok(line, " ");
3759f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      while (s) {
3760f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        args[i] = s;
3761f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        ++i;
3762f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        s = strtok(0, " ");
3763f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      }
3764f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (print_usrs(&args[0], &args[i]))
3765f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        return 1;
3766f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
3767f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    else
3768f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      line[numChars++] = c;
3769f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3770f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3771f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fclose(fp);
3772f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 0;
3773f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3774f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3775f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/******************************************************************************/
37760d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Command line processing.                                                   */
37770d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
37787ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregorint write_pch_file(const char *filename, int argc, const char *argv[]) {
37797ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  CXIndex Idx;
37807ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  CXTranslationUnit TU;
37817ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
37827ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  int num_unsaved_files = 0;
3783651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
378408aa622ab923fb8a2751f213feee0b9124ee86fbFrancois Pichet  int result = 0;
37857ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
3786fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit  Idx = clang_createIndex(/* excludeDeclsFromPCH */1, /* displayDiagnostics=*/1);
37877ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
37887ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
37897ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    clang_disposeIndex(Idx);
37907ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return -1;
37917ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  }
3792651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
3793651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(
3794651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Idx, 0, argv + num_unsaved_files, argc - num_unsaved_files,
3795651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      unsaved_files, num_unsaved_files,
3796651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      CXTranslationUnit_Incomplete |
3797651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          CXTranslationUnit_DetailedPreprocessingRecord |
3798651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          CXTranslationUnit_ForSerialization,
3799651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      &TU);
3800651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
38017ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
3802651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
38037ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
3804651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_disposeTranslationUnit(TU);
38057ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    clang_disposeIndex(Idx);
38067ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return 1;
38077ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  }
38087ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
380939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  switch (clang_saveTranslationUnit(TU, filename,
381039c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor                                    clang_defaultSaveOptions(TU))) {
381139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_None:
381239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
381339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
381439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_TranslationErrors:
381539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: translation errors\n",
381639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor            filename);
381739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 2;
381839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
381939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
382039c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_InvalidTU:
382139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: invalid translation unit\n",
382239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor            filename);
382339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 3;
382439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
382539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
382639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_Unknown:
382739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  default:
382839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: unknown error \n", filename);
382939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 1;
383039c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
383139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  }
383239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
38337ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  clang_disposeTranslationUnit(TU);
38347ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
38357ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  clang_disposeIndex(Idx);
383639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  return result;
38377ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor}
38387ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
38397ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/******************************************************************************/
3840153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/* Serialized diagnostics.                                                    */
3841153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/******************************************************************************/
3842153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3843153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic const char *getDiagnosticCodeStr(enum CXLoadDiag_Error error) {
3844153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  switch (error) {
3845153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_CannotLoad: return "Cannot Load File";
3846153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_None: break;
3847153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_Unknown: return "Unknown";
3848153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_InvalidFile: return "Invalid File";
3849153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3850153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return "None";
3851153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3852153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3853153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic const char *getSeverityString(enum CXDiagnosticSeverity severity) {
3854153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  switch (severity) {
3855153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Note: return "note";
3856153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Error: return "error";
3857153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Fatal: return "fatal";
3858153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Ignored: return "ignored";
3859153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Warning: return "warning";
3860153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3861153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return "unknown";
3862153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3863153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3864153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printIndent(unsigned indent) {
3865a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  if (indent == 0)
3866a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek    return;
3867a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  fprintf(stderr, "+");
3868a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  --indent;
3869153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  while (indent > 0) {
3870a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek    fprintf(stderr, "-");
3871153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    --indent;
3872153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3873153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3874153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3875153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printLocation(CXSourceLocation L) {
3876153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXFile File;
3877153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXString FileName;
3878153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned line, column, offset;
3879153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3880153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_getExpansionLocation(L, &File, &line, &column, &offset);
3881153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  FileName = clang_getFileName(File);
3882153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3883153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  fprintf(stderr, "%s:%d:%d", clang_getCString(FileName), line, column);
3884153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_disposeString(FileName);
3885153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3886153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3887153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printRanges(CXDiagnostic D, unsigned indent) {
3888153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned i, n = clang_getDiagnosticNumRanges(D);
3889153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3890153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0; i < n; ++i) {
3891153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceLocation Start, End;
389210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceRange SR = clang_getDiagnosticRange(D, i);
3893153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    Start = clang_getRangeStart(SR);
3894153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    End = clang_getRangeEnd(SR);
3895153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3896153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3897153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "Range: ");
3898153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(Start);
3899153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, " ");
3900153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(End);
3901153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "\n");
3902153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3903153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3904153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3905153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printFixIts(CXDiagnostic D, unsigned indent) {
3906153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned i, n = clang_getDiagnosticNumFixIts(D);
39073739b32a01b9778176757425e62252c288b85383Ted Kremenek  fprintf(stderr, "Number FIXITs = %d\n", n);
3908153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0 ; i < n; ++i) {
3909153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceRange ReplacementRange;
3910153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXString text;
3911153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    text = clang_getDiagnosticFixIt(D, i, &ReplacementRange);
3912153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3913153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3914153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "FIXIT: (");
3915153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(clang_getRangeStart(ReplacementRange));
3916153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, " - ");
3917153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(clang_getRangeEnd(ReplacementRange));
3918153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "): \"%s\"\n", clang_getCString(text));
3919153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(text);
3920153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3921153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3922153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3923153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printDiagnosticSet(CXDiagnosticSet Diags, unsigned indent) {
39249190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi  unsigned i, n;
39259190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi
3926153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (!Diags)
3927153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    return;
3928153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
39299190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi  n = clang_getNumDiagnosticsInSet(Diags);
3930153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0; i < n; ++i) {
3931153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceLocation DiagLoc;
3932153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXDiagnostic D;
3933153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXFile File;
393478d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    CXString FileName, DiagSpelling, DiagOption, DiagCat;
3935153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    unsigned line, column, offset;
393678d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    const char *DiagOptionStr = 0, *DiagCatStr = 0;
3937153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3938153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    D = clang_getDiagnosticInSet(Diags, i);
3939153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagLoc = clang_getDiagnosticLocation(D);
3940153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_getExpansionLocation(DiagLoc, &File, &line, &column, &offset);
3941153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    FileName = clang_getFileName(File);
3942153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagSpelling = clang_getDiagnosticSpelling(D);
3943153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3944153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3945153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3946153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "%s:%d:%d: %s: %s",
3947153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(FileName),
3948153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            line,
3949153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            column,
3950153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            getSeverityString(clang_getDiagnosticSeverity(D)),
3951153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(DiagSpelling));
3952153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3953153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagOption = clang_getDiagnosticOption(D, 0);
3954153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagOptionStr = clang_getCString(DiagOption);
3955153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    if (DiagOptionStr) {
3956153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek      fprintf(stderr, " [%s]", DiagOptionStr);
3957153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    }
3958153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
395978d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    DiagCat = clang_getDiagnosticCategoryText(D);
396078d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    DiagCatStr = clang_getCString(DiagCat);
396178d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    if (DiagCatStr) {
396278d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek      fprintf(stderr, " [%s]", DiagCatStr);
396378d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    }
396478d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek
3965153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "\n");
3966153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3967153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printRanges(D, indent);
3968153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printFixIts(D, indent);
3969153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3970a4ca95a6ebe85d63df08f3290372e97ddaf0b02aNAKAMURA Takumi    /* Print subdiagnostics. */
3971153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printDiagnosticSet(clang_getChildDiagnostics(D), indent+2);
3972153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3973153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(FileName);
3974153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(DiagSpelling);
3975153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(DiagOption);
39766bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    clang_disposeString(DiagCat);
3977153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3978153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3979153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3980153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic int read_diagnostics(const char *filename) {
3981153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  enum CXLoadDiag_Error error;
3982153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXString errorString;
3983153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXDiagnosticSet Diags = 0;
3984153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3985153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  Diags = clang_loadDiagnostics(filename, &error, &errorString);
3986153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (!Diags) {
3987153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "Trouble deserializing file (%s): %s\n",
3988153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            getDiagnosticCodeStr(error),
3989153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(errorString));
3990153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(errorString);
3991153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    return 1;
3992153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3993153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3994153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  printDiagnosticSet(Diags, 0);
3995a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  fprintf(stderr, "Number of diagnostics: %d\n",
3996a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek          clang_getNumDiagnosticsInSet(Diags));
3997153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_disposeDiagnosticSet(Diags);
3998153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return 0;
3999153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
4000153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
4001651f13cea278ec967336033dd032faef0e9fc2ecStephen Hinesstatic int perform_print_build_session_timestamp(void) {
4002651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  printf("%lld\n", clang_getBuildSessionTimestamp());
4003651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  return 0;
4004651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines}
4005651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
4006153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/******************************************************************************/
40077ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/* Command line processing.                                                   */
40087ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/******************************************************************************/
4009f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek
4010e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregorstatic CXCursorVisitor GetVisitor(const char *s) {
40117d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  if (s[0] == '\0')
4012e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return FilteredPrintingVisitor;
40137d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  if (strcmp(s, "-usrs") == 0)
40147d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    return USRVisitor;
401559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  if (strncmp(s, "-memory-usage", 13) == 0)
401659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    return GetVisitor(s + 13);
40177d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  return NULL;
40187d40562f83552b7295411e10ee887d8d55470679Ted Kremenek}
40197d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
4020f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenekstatic void print_usage(void) {
4021f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  fprintf(stderr,
40220d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "usage: c-index-test -code-completion-at=<site> <compiler arguments>\n"
40231982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -code-completion-timing=<site> <compiler arguments>\n"
4024f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    "       c-index-test -cursor-at=<site> <compiler arguments>\n"
4025ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    "       c-index-test -file-refs-at=<site> <compiler arguments>\n"
4026ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    "       c-index-test -file-includes-in=<filename> <compiler arguments>\n");
40273584972866f39d42aa4029586278022d89451bd9NAKAMURA Takumi  fprintf(stderr,
40284e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    "       c-index-test -index-file [-check-prefix=<FileCheck prefix>] <compiler arguments>\n"
402911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    "       c-index-test -index-file-full [-check-prefix=<FileCheck prefix>] <compiler arguments>\n"
403021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    "       c-index-test -index-tu [-check-prefix=<FileCheck prefix>] <AST file>\n"
4031d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    "       c-index-test -index-compile-db [-check-prefix=<FileCheck prefix>] <compilation database>\n"
40321d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    "       c-index-test -test-file-scan <AST file> <source file> "
403326fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen          "[FileCheck prefix]\n");
403426fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen  fprintf(stderr,
4035fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek    "       c-index-test -test-load-tu <AST file> <symbol filter> "
4036fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek          "[FileCheck prefix]\n"
40377d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    "       c-index-test -test-load-tu-usrs <AST file> <symbol filter> "
40387d40562f83552b7295411e10ee887d8d55470679Ted Kremenek           "[FileCheck prefix]\n"
40391982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -test-load-source <symbol filter> {<args>}*\n");
4040f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  fprintf(stderr,
404159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-load-source-memory-usage "
404259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "<symbol filter> {<args>}*\n"
4043abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    "       c-index-test -test-load-source-reparse <trials> <symbol filter> "
4044abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    "          {<args>}*\n"
40451982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -test-load-source-usrs <symbol filter> {<args>}*\n"
404659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-load-source-usrs-memory-usage "
404759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek          "<symbol filter> {<args>}*\n"
404816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    "       c-index-test -test-annotate-tokens=<range> {<args>}*\n"
404916b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    "       c-index-test -test-inclusion-stack-source {<args>}*\n"
40504e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    "       c-index-test -test-inclusion-stack-tu <AST file>\n");
405153513d2e24205965ffd5d4780a10f41cc335f17bChandler Carruth  fprintf(stderr,
40524e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    "       c-index-test -test-print-linkage-source {<args>}*\n"
4053ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko    "       c-index-test -test-print-type {<args>}*\n"
4054411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    "       c-index-test -test-print-type-size {<args>}*\n"
40551eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    "       c-index-test -test-print-bitwidth {<args>}*\n"
405659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -print-usr [<CursorKind> {<args>}]*\n"
40577ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    "       c-index-test -print-usr-file <file>\n"
4058153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    "       c-index-test -write-pch <file> <compiler arguments>\n");
4059153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  fprintf(stderr,
4060db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    "       c-index-test -compilation-db [lookup <filename>] database\n");
4061db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  fprintf(stderr,
4062651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    "       c-index-test -print-build-session-timestamp\n");
4063651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  fprintf(stderr,
4064153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    "       c-index-test -read-diagnostics <file>\n\n");
4065caf4bd3d3c66fd95b0159356e2c4da7fbf2897f3Douglas Gregor  fprintf(stderr,
40667d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    " <symbol filter> values:\n%s",
40670d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   all - load all symbols, including those from PCH\n"
40680d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   local - load all symbols except those in PCH\n"
40690d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   category - only load ObjC categories (non-PCH)\n"
40700d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   interface - only load ObjC interfaces (non-PCH)\n"
40710d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   protocol - only load ObjC protocols (non-PCH)\n"
40720d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   function - only load functions (non-PCH)\n"
4073625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    "   typedef - only load typdefs (non-PCH)\n"
4074625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    "   scan-function - scan function bodies (non-PCH)\n\n");
4075f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek}
4076f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek
40776edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/***/
40786edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
40796edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbarint cindextest_main(int argc, const char **argv) {
40800a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  clang_enableStackTraces();
4081153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (argc > 2 && strcmp(argv[1], "-read-diagnostics") == 0)
4082153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek      return read_diagnostics(argv[2]);
4083f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  if (argc > 2 && strstr(argv[1], "-code-completion-at=") == argv[1])
40841982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    return perform_code_completion(argc, argv, 0);
40851982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  if (argc > 2 && strstr(argv[1], "-code-completion-timing=") == argv[1])
40861982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    return perform_code_completion(argc, argv, 1);
4087f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  if (argc > 2 && strstr(argv[1], "-cursor-at=") == argv[1])
4088f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return inspect_cursor_at(argc, argv);
4089aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (argc > 2 && strstr(argv[1], "-file-refs-at=") == argv[1])
4090aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return find_file_refs_at(argc, argv);
4091ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  if (argc > 2 && strstr(argv[1], "-file-includes-in=") == argv[1])
4092ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    return find_file_includes_in(argc, argv);
40934e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-file") == 0)
409411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    return index_file(argc - 2, argv + 2, /*full=*/0);
409511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-file-full") == 0)
409611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    return index_file(argc - 2, argv + 2, /*full=*/1);
409721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-tu") == 0)
409821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return index_tu(argc - 2, argv + 2);
4099d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-compile-db") == 0)
4100d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return index_compile_db(argc - 2, argv + 2);
41017d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strncmp(argv[1], "-test-load-tu", 13) == 0) {
4102e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 13);
41037d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    if (I)
4104ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek      return perform_test_load_tu(argv[2], argv[3], argc >= 5 ? argv[4] : 0, I,
4105ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                                  NULL);
41067d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  }
4107abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  else if (argc >= 5 && strncmp(argv[1], "-test-load-source-reparse", 25) == 0){
4108abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 25);
4109abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    if (I) {
4110abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      int trials = atoi(argv[2]);
4111abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      return perform_test_reparse_source(argc - 4, argv + 4, trials, argv[3], I,
4112abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                         NULL);
4113abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    }
4114abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
41157d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strncmp(argv[1], "-test-load-source", 17) == 0) {
4116e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 17);
411759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
411859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    PostVisitTU postVisit = 0;
411959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    if (strstr(argv[1], "-memory-usage"))
412059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      postVisit = PrintMemoryUsage;
412159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
41227d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    if (I)
412359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      return perform_test_load_source(argc - 3, argv + 3, argv[2], I,
412459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek                                      postVisit);
41257d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  }
41267d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strcmp(argv[1], "-test-file-scan") == 0)
41271d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    return perform_file_scan(argv[2], argv[3],
41281d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                             argc >= 5 ? argv[4] : 0);
4129fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  else if (argc > 2 && strstr(argv[1], "-test-annotate-tokens=") == argv[1])
4130fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return perform_token_annotation(argc, argv);
413116b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-inclusion-stack-source") == 0)
413216b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all", NULL,
413316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                                    PrintInclusionStack);
413416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-inclusion-stack-tu") == 0)
413516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    return perform_test_load_tu(argv[2], "all", NULL, NULL,
413616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                                PrintInclusionStack);
41373bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-print-linkage-source") == 0)
41383bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all", PrintLinkage,
41393bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek                                    NULL);
4140ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  else if (argc > 2 && strcmp(argv[1], "-test-print-type") == 0)
41418e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all",
4142ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko                                    PrintType, 0);
4143411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  else if (argc > 2 && strcmp(argv[1], "-test-print-type-size") == 0)
4144411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    return perform_test_load_source(argc - 2, argv + 2, "all",
4145411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis                                    PrintTypeSize, 0);
41461eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  else if (argc > 2 && strcmp(argv[1], "-test-print-bitwidth") == 0)
41471eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    return perform_test_load_source(argc - 2, argv + 2, "all",
41481eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko                                    PrintBitWidth, 0);
4149176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines  else if (argc > 2 && strcmp(argv[1], "-test-print-mangle") == 0)
4150176edba5311f6eff0cad2631449885ddf4fbc9eaStephen Hines    return perform_test_load_tu(argv[2], "all", NULL, PrintMangledName, NULL);
4151f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  else if (argc > 1 && strcmp(argv[1], "-print-usr") == 0) {
4152f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    if (argc > 2)
4153f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      return print_usrs(argv + 2, argv + argc);
4154f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    else {
4155f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      display_usrs();
4156f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      return 1;
4157f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
4158f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
4159f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-print-usr-file") == 0)
4160f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return print_usrs_file(argv[2]);
41617ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  else if (argc > 2 && strcmp(argv[1], "-write-pch") == 0)
41627ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return write_pch_file(argv[2], argc - 3, argv + 3);
4163db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  else if (argc > 2 && strcmp(argv[1], "-compilation-db") == 0)
4164db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    return perform_test_compilation_db(argv[argc-1], argc - 3, argv + 2);
4165651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  else if (argc == 2 && strcmp(argv[1], "-print-build-session-timestamp") == 0)
4166651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return perform_print_build_session_timestamp();
4167db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
4168f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  print_usage();
416950398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff  return 1;
417050398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff}
41716edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
41726edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/***/
41736edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
41746edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/* We intentionally run in a separate thread to ensure we at least minimal
41756edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar * testing of a multithreaded environment (for example, having a reduced stack
41766edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar * size). */
41776edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
41786edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbartypedef struct thread_info {
41796edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  int argc;
41806edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  const char **argv;
41816edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  int result;
41826edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar} thread_info;
418384294911b83003bcac726d00ffa91143752feb57Benjamin Kramervoid thread_runner(void *client_data_v) {
41846edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  thread_info *client_data = client_data_v;
41856edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data->result = cindextest_main(client_data->argc, client_data->argv);
4186c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines}
4187c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines
4188c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hinesstatic void flush_atexit(void) {
4189c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines  /* stdout, and surprisingly even stderr, are not always flushed on process
4190c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines   * and thread exit, particularly when the system is under heavy load. */
4191c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines  fflush(stdout);
4192c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines  fflush(stderr);
41936edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar}
41946edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
41956edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbarint main(int argc, const char **argv) {
4196d1a4f68a4301d1ee3098cc9db0cd507b96dd1beeBenjamin Kramer  thread_info client_data;
4197d1a4f68a4301d1ee3098cc9db0cd507b96dd1beeBenjamin Kramer
4198c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines  atexit(flush_atexit);
4199c568f1e98938584c0ef0b12ae5018ff7d90a4072Stephen Hines
4200f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
4201f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  LIBXML_TEST_VERSION
4202f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
4203f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
42046160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor  if (getenv("CINDEXTEST_NOTHREADS"))
42056160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor    return cindextest_main(argc, argv);
42066160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor
42076edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data.argc = argc;
42086edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data.argv = argv;
4209a32a6e1abe6972b48b4d5ec79d81b6f7241723bfDaniel Dunbar  clang_executeOnThread(thread_runner, &client_data, 0);
42106edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  return client_data.result;
42116edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar}
4212