12b8ee6c2994f738e5162ff46b638974870f51662Steve Naroff/* c-index-test.c */
250398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
3ef8225444452a1486bd721f3285301fe84643b00Stephen 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    }
799e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor
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);
805e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      printf(" [Specialization of %s:%d:%d]",
806e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor             clang_getCString(Name), line, column);
807e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      clang_disposeString(Name);
808e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    }
8099f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
8109f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    clang_getOverriddenCursors(Cursor, &overridden, &num_overridden);
8119f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    if (num_overridden) {
8129f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      unsigned I;
813b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      LineCol lineCols[50];
814b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      assert(num_overridden <= 50);
8159f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      printf(" [Overrides ");
8169f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      for (I = 0; I != num_overridden; ++I) {
81710f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        CXSourceLocation Loc = clang_getCursorLocation(overridden[I]);
818a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor        clang_getSpellingLocation(Loc, 0, &line, &column, 0);
819b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        lineCols[I].line = line;
820b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        lineCols[I].col = column;
821b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      }
822642214936430d2258318f2022184c1709dfa16d3Michael Liao      /* Make the order of the override list deterministic. */
823b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      qsort(lineCols, num_overridden, sizeof(LineCol), lineCol_cmp);
824b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      for (I = 0; I != num_overridden; ++I) {
8259f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor        if (I)
8269f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor          printf(", ");
827b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        printf("@%d:%d", lineCols[I].line, lineCols[I].col);
8289f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      }
8299f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      printf("]");
8309f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      clang_disposeOverriddenCursors(overridden);
8319f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    }
832ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
833ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    if (Cursor.kind == CXCursor_InclusionDirective) {
83410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXFile File = clang_getIncludedFile(Cursor);
83510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXString Included = clang_getFileName(File);
836ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      printf(" (%s)", clang_getCString(Included));
837ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      clang_disposeString(Included);
838dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor
839dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor      if (clang_isFileMultipleIncludeGuarded(TU, File))
840dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor        printf("  [multi-include guarded]");
841ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    }
842430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
843430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CursorExtent = clang_getCursorExtent(Cursor);
844430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    RefNameRange = clang_getCursorReferenceNameRange(Cursor,
845430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                   CXNameRange_WantQualifier
846430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantSinglePiece
847430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantTemplateArgs,
848430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                     0);
849430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (!clang_equalRanges(CursorExtent, RefNameRange))
850430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      PrintRange(RefNameRange, "SingleRefName");
851430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
852430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    for (RefNameRangeNr = 0; 1; RefNameRangeNr++) {
853430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      RefNameRange = clang_getCursorReferenceNameRange(Cursor,
854430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                   CXNameRange_WantQualifier
855430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantTemplateArgs,
856430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                       RefNameRangeNr);
857430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (clang_equalRanges(clang_getNullRange(), RefNameRange))
858430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        break;
859430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (!clang_equalRanges(CursorExtent, RefNameRange))
860430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        PrintRange(RefNameRange, "RefName");
861430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    }
862aa0cd85838f2a024e589ea4e8c2094130065af21Dmitri Gribenko
8636bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    PrintCursorComments(Cursor, CommentSchemaFile);
8649ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis
8659ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis    {
8669ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis      unsigned PropAttrs = clang_Cursor_getObjCPropertyAttributes(Cursor, 0);
8679ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis      if (PropAttrs != CXObjCPropertyAttr_noattr) {
8689ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        printf(" [");
8699ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        #define PRINT_PROP_ATTR(A) \
8709ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis          if (PropAttrs & CXObjCPropertyAttr_##A) printf(#A ",")
8719ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(readonly);
8729ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(getter);
8739ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(assign);
8749ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(readwrite);
8759ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(retain);
8769ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(copy);
8779ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(nonatomic);
8789ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(setter);
8799ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(atomic);
8809ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(weak);
8819ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(strong);
8829ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        PRINT_PROP_ATTR(unsafe_unretained);
8839ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis        printf("]");
8849ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis      }
8859ee6a66d19aacc35397bbdc4fd535e3944301856Argyrios Kyrtzidis    }
88638dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis
88738dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis    {
88838dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis      unsigned QT = clang_Cursor_getObjCDeclQualifiers(Cursor);
88938dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis      if (QT != CXObjCDeclQualifier_None) {
89038dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        printf(" [");
89138dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        #define PRINT_OBJC_QUAL(A) \
89238dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis          if (QT & CXObjCDeclQualifier_##A) printf(#A ",")
89338dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(In);
89438dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Inout);
89538dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Out);
89638dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Bycopy);
89738dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Byref);
89838dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        PRINT_OBJC_QUAL(Oneway);
89938dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis        printf("]");
90038dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis      }
90138dbad233bfc7906e38e028707ffe5925b9ca46cArgyrios Kyrtzidis    }
902699a07d8a0b1579c5178b3baf4bcf9edb6b38108Steve Naroff  }
903af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff}
90489922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff
905e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekstatic const char* GetCursorSource(CXCursor Cursor) {
90610f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXSourceLocation Loc = clang_getCursorLocation(Cursor);
90774844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  CXString source;
9081db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  CXFile file;
909b4efaa0a14dd2382aa028c03283b5a7f5345e24dArgyrios Kyrtzidis  clang_getExpansionLocation(Loc, &file, 0, 0, 0);
9101db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  source = clang_getFileName(file);
911e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (!clang_getCString(source)) {
91274844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
91374844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    return "<invalid loc>";
91474844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  }
91574844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  else {
916e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    const char *b = basename(clang_getCString(source));
91774844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
91874844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    return b;
91974844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  }
9209298cfc7475c48fa42b168c37f628663d65ddde7Ted Kremenek}
9219298cfc7475c48fa42b168c37f628663d65ddde7Ted Kremenek
9220d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
923ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/* Callbacks.                                                                 */
924ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/******************************************************************************/
925ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek
926ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenektypedef void (*PostVisitTU)(CXTranslationUnit);
927ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek
928a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregorvoid PrintDiagnostic(CXDiagnostic Diagnostic) {
929a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  FILE *out = stderr;
9305352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  CXFile file;
931274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  CXString Msg;
9320a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  unsigned display_opts = CXDiagnostic_DisplaySourceLocation
933aa5f135f8db82b5e5fb1640fd51f8078e0b2d82dDouglas Gregor    | CXDiagnostic_DisplayColumn | CXDiagnostic_DisplaySourceRanges
934aa5f135f8db82b5e5fb1640fd51f8078e0b2d82dDouglas Gregor    | CXDiagnostic_DisplayOption;
9350a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  unsigned i, num_fixits;
936f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
9370a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (clang_getDiagnosticSeverity(Diagnostic) == CXDiagnostic_Ignored)
9385352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return;
939e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
940274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  Msg = clang_formatDiagnostic(Diagnostic, display_opts);
941274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  fprintf(stderr, "%s\n", clang_getCString(Msg));
942274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  clang_disposeString(Msg);
943f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
944a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  clang_getSpellingLocation(clang_getDiagnosticLocation(Diagnostic),
945a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                            &file, 0, 0, 0);
9460a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (!file)
9470a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    return;
948e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
9490a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  num_fixits = clang_getDiagnosticNumFixIts(Diagnostic);
9503739b32a01b9778176757425e62252c288b85383Ted Kremenek  fprintf(stderr, "Number FIX-ITs = %d\n", num_fixits);
9510a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  for (i = 0; i != num_fixits; ++i) {
952473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXSourceRange range;
95310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString insertion_text = clang_getDiagnosticFixIt(Diagnostic, i, &range);
95410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceLocation start = clang_getRangeStart(range);
95510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceLocation end = clang_getRangeEnd(range);
956473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    unsigned start_line, start_column, end_line, end_column;
957473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXFile start_file, end_file;
958a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(start, &start_file, &start_line,
959a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              &start_column, 0);
960a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(end, &end_file, &end_line, &end_column, 0);
961473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    if (clang_equalLocations(start, end)) {
962473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Insertion. */
963473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      if (start_file == file)
9640a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Insert \"%s\" at %d:%d\n",
965473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor                clang_getCString(insertion_text), start_line, start_column);
966473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    } else if (strcmp(clang_getCString(insertion_text), "") == 0) {
967473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Removal. */
9680a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      if (start_file == file && end_file == file) {
9690a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Remove ");
9700a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        PrintExtent(out, start_line, start_column, end_line, end_column);
9710a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "\n");
972436f3f0400c633251e4071f81358c47bab964adfDouglas Gregor      }
973473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    } else {
974473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Replacement. */
9750a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      if (start_file == end_file) {
9760a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Replace ");
9770a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        PrintExtent(out, start_line, start_column, end_line, end_column);
978473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor        fprintf(out, " with \"%s\"\n", clang_getCString(insertion_text));
97951c6d384551674facc19f745ecf6e289d28dc55fDouglas Gregor      }
9800a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    }
981473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    clang_disposeString(insertion_text);
98251c6d384551674facc19f745ecf6e289d28dc55fDouglas Gregor  }
9835352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor}
9845352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor
9857473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenekvoid PrintDiagnosticSet(CXDiagnosticSet Set) {
9867473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  int i = 0, n = clang_getNumDiagnosticsInSet(Set);
9877473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  for ( ; i != n ; ++i) {
9887473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    CXDiagnostic Diag = clang_getDiagnosticInSet(Set, i);
9897473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    CXDiagnosticSet ChildDiags = clang_getChildDiagnostics(Diag);
990a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    PrintDiagnostic(Diag);
9917473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    if (ChildDiags)
9927473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek      PrintDiagnosticSet(ChildDiags);
9937473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  }
9947473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek}
9957473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek
9967473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenekvoid PrintDiagnostics(CXTranslationUnit TU) {
9977473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  CXDiagnosticSet TUSet = clang_getDiagnosticSetFromTU(TU);
9987473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  PrintDiagnosticSet(TUSet);
9997473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  clang_disposeDiagnosticSet(TUSet);
1000a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor}
1001a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor
100259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenekvoid PrintMemoryUsage(CXTranslationUnit TU) {
1003b22732365ca26fbccacc4bdd32e875342f571e94Matt Beaumont-Gay  unsigned long total = 0;
10044e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  unsigned i = 0;
100510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXTUResourceUsage usage = clang_getCXTUResourceUsage(TU);
10063c683367073e2d98a9046060f9bc7db872a1c63dFrancois Pichet  fprintf(stderr, "Memory usage:\n");
10074e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  for (i = 0 ; i != usage.numEntries; ++i) {
1008f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    const char *name = clang_getTUResourceUsageName(usage.entries[i].kind);
100959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    unsigned long amount = usage.entries[i].amount;
101059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    total += amount;
10114e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    fprintf(stderr, "  %s : %ld bytes (%f MBytes)\n", name, amount,
101259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek            ((double) amount)/(1024*1024));
101359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  }
10144e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  fprintf(stderr, "  TOTAL = %ld bytes (%f MBytes)\n", total,
101559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek          ((double) total)/(1024*1024));
1016f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  clang_disposeCXTUResourceUsage(usage);
101759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
101859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
1019ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/******************************************************************************/
1020e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor/* Logic for testing traversal.                                               */
10210d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
10220d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
1023a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregorstatic void PrintCursorExtent(CXCursor C) {
102410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXSourceRange extent = clang_getCursorExtent(C);
102510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  PrintRange(extent, "Extent");
1026fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek}
1027fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek
1028f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko/* Data used by the visitors. */
1029f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkotypedef struct {
1030e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  CXTranslationUnit TU;
1031e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  enum CXCursorKind *Filter;
10326bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  const char *CommentSchemaFile;
1033e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor} VisitorData;
1034fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek
1035625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1036e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekenum CXChildVisitResult FilteredPrintingVisitor(CXCursor Cursor,
1037e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                CXCursor Parent,
1038e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                CXClientData ClientData) {
1039e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
1040e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  if (!Data->Filter || (Cursor.kind == *(enum CXCursorKind *)Data->Filter)) {
104110f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceLocation Loc = clang_getCursorLocation(Cursor);
10421db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    unsigned line, column;
1043a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(Loc, 0, &line, &column, 0);
1044fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek    printf("// %s: %s:%d:%d: ", FileCheckPrefix,
10451db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor           GetCursorSource(Cursor), line, column);
10466bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    PrintCursor(Cursor, Data->CommentSchemaFile);
1047a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    PrintCursorExtent(Cursor);
104804b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis    if (clang_isDeclaration(Cursor.kind)) {
104904b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      enum CX_CXXAccessSpecifier access = clang_getCXXAccessSpecifier(Cursor);
105004b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      const char *accessStr = 0;
105104b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis
105204b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      switch (access) {
105304b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        case CX_CXXInvalidAccessSpecifier: break;
105404b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        case CX_CXXPublic:
105504b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis          accessStr = "public"; break;
105604b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        case CX_CXXProtected:
105704b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis          accessStr = "protected"; break;
105804b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        case CX_CXXPrivate:
105904b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis          accessStr = "private"; break;
106004b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      }
106104b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis
106204b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis      if (accessStr)
106304b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis        printf(" [access=%s]", accessStr);
106404b6748ee90649069ee79464317ae6cd6aca317dArgyrios Kyrtzidis    }
1065e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("\n");
1066e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Recurse;
1067625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  }
1068e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1069e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
1070625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar}
1071c857ea4d22e1e6dd9ede1f0e84d48b157c6924fdSteve Naroff
1072e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekstatic enum CXChildVisitResult FunctionScanVisitor(CXCursor Cursor,
1073e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                   CXCursor Parent,
1074e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                   CXClientData ClientData) {
1075625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  const char *startBuf, *endBuf;
1076625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  unsigned startLine, startColumn, endLine, endColumn, curLine, curColumn;
1077625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXCursor Ref;
1078e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
1079625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1080b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (Cursor.kind != CXCursor_FunctionDecl ||
1081b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      !clang_isCursorDefinition(Cursor))
1082e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Continue;
1083625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1084625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  clang_getDefinitionSpellingAndExtent(Cursor, &startBuf, &endBuf,
1085625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar                                       &startLine, &startColumn,
1086625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar                                       &endLine, &endColumn);
1087625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  /* Probe the entire body, looking for both decls and refs. */
1088625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  curLine = startLine;
1089625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  curColumn = startColumn;
1090625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1091625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  while (startBuf < endBuf) {
109298258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    CXSourceLocation Loc;
10931db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    CXFile file;
109474844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    CXString source;
1095e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1096625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    if (*startBuf == '\n') {
1097625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      startBuf++;
1098625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curLine++;
1099625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curColumn = 1;
1100625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    } else if (*startBuf != '\t')
1101625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curColumn++;
1102e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
110398258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    Loc = clang_getCursorLocation(Cursor);
1104a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(Loc, &file, 0, 0, 0);
1105e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
11061db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    source = clang_getFileName(file);
1107e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    if (clang_getCString(source)) {
110810f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXSourceLocation RefLoc
110910f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        = clang_getLocation(Data->TU, file, curLine, curColumn);
1110b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor      Ref = clang_getCursor(Data->TU, RefLoc);
111198258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      if (Ref.kind == CXCursor_NoDeclFound) {
111298258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        /* Nothing found here; that's fine. */
111398258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      } else if (Ref.kind != CXCursor_FunctionDecl) {
111498258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        printf("// %s: %s:%d:%d: ", FileCheckPrefix, GetCursorSource(Ref),
111598258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor               curLine, curColumn);
11166bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        PrintCursor(Ref, Data->CommentSchemaFile);
111798258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        printf("\n");
111898258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      }
11194ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff    }
112074844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
1121625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    startBuf++;
11222d4d629d8a0de5112c7ae9d05c03ddbf6dcd956aSteve Naroff  }
1123e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1124e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
112589922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff}
112650398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
11277d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
11287d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/* USR testing.                                                               */
11297d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
11307d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
1131e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregorenum CXChildVisitResult USRVisitor(CXCursor C, CXCursor parent,
1132e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                   CXClientData ClientData) {
1133e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
1134e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  if (!Data->Filter || (C.kind == *(enum CXCursorKind *)Data->Filter)) {
113510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString USR = clang_getCursorUSR(C);
113610f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    const char *cstr = clang_getCString(USR);
1137e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    if (!cstr || cstr[0] == '\0') {
11387d40562f83552b7295411e10ee887d8d55470679Ted Kremenek      clang_disposeString(USR);
1139e74ef1289d5fff0a6ea573198bf354fa8cd84d51Ted Kremenek      return CXChildVisit_Recurse;
11407d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    }
1141e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    printf("// %s: %s %s", FileCheckPrefix, GetCursorSource(C), cstr);
1142e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek
1143a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    PrintCursorExtent(C);
11447d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    printf("\n");
11457d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    clang_disposeString(USR);
1146e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1147e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Recurse;
1148e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
1149e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1150e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
11517d40562f83552b7295411e10ee887d8d55470679Ted Kremenek}
11527d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
11537d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
115416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/* Inclusion stack testing.                                                   */
115516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/******************************************************************************/
115616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
115716b55a71695a33c094383295cc7b7a2080e098daTed Kremenekvoid InclusionVisitor(CXFile includedFile, CXSourceLocation *includeStack,
115816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                      unsigned includeStackLen, CXClientData data) {
1159e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
116016b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  unsigned i;
116174844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  CXString fname;
116274844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek
116374844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  fname = clang_getFileName(includedFile);
1164e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  printf("file: %s\nincluded by:\n", clang_getCString(fname));
116574844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  clang_disposeString(fname);
1166e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
116716b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  for (i = 0; i < includeStackLen; ++i) {
116816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    CXFile includingFile;
116916b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    unsigned line, column;
1170a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(includeStack[i], &includingFile, &line,
1171a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              &column, 0);
117274844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    fname = clang_getFileName(includingFile);
1173e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("  %s:%d:%d\n", clang_getCString(fname), line, column);
117474844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(fname);
117516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  }
117616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  printf("\n");
117716b55a71695a33c094383295cc7b7a2080e098daTed Kremenek}
117816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
117916b55a71695a33c094383295cc7b7a2080e098daTed Kremenekvoid PrintInclusionStack(CXTranslationUnit TU) {
1180e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  clang_getInclusions(TU, InclusionVisitor, NULL);
118116b55a71695a33c094383295cc7b7a2080e098daTed Kremenek}
118216b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
118316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/******************************************************************************/
11843bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/* Linkage testing.                                                           */
11853bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/******************************************************************************/
11863bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
11873bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenekstatic enum CXChildVisitResult PrintLinkage(CXCursor cursor, CXCursor p,
11883bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek                                            CXClientData d) {
11893bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  const char *linkage = 0;
11903bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
11913bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  if (clang_isInvalid(clang_getCursorKind(cursor)))
11923bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    return CXChildVisit_Recurse;
11933bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
11943bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  switch (clang_getCursorLinkage(cursor)) {
11953bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    case CXLinkage_Invalid: break;
1196c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_NoLinkage: linkage = "NoLinkage"; break;
1197c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_Internal: linkage = "Internal"; break;
1198c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_UniqueExternal: linkage = "UniqueExternal"; break;
1199c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_External: linkage = "External"; break;
12003bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  }
12013bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12023bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  if (linkage) {
1203f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(cursor, NULL);
12043bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    printf("linkage=%s\n", linkage);
12053bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  }
12063bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12073bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  return CXChildVisit_Recurse;
12083bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek}
12093bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
12103bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/******************************************************************************/
12118e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/* Typekind testing.                                                          */
12128e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/******************************************************************************/
12138e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
1214ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenkostatic void PrintTypeAndTypeKind(CXType T, const char *Format) {
1215ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  CXString TypeSpelling, TypeKindSpelling;
1216ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko
1217ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  TypeSpelling = clang_getTypeSpelling(T);
1218ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  TypeKindSpelling = clang_getTypeKindSpelling(T.kind);
1219ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  printf(Format,
1220ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko         clang_getCString(TypeSpelling),
1221ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko         clang_getCString(TypeKindSpelling));
1222ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  clang_disposeString(TypeSpelling);
1223ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  clang_disposeString(TypeKindSpelling);
1224ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko}
1225ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko
1226ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenkostatic enum CXChildVisitResult PrintType(CXCursor cursor, CXCursor p,
1227ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko                                         CXClientData d) {
12288e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  if (!clang_isInvalid(clang_getCursorKind(cursor))) {
122910f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXType T = clang_getCursorType(cursor);
1230659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis    enum CXRefQualifierKind RQ = clang_Type_getCXXRefQualifier(T);
1231f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(cursor, NULL);
1232ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko    PrintTypeAndTypeKind(T, " [type=%s] [typekind=%s]");
1233e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isConstQualifiedType(T))
1234e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" const");
1235e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isVolatileQualifiedType(T))
1236e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" volatile");
1237e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isRestrictQualifiedType(T))
1238e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" restrict");
1239659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis    if (RQ == CXRefQualifier_LValue)
1240659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis      printf(" lvalue-ref-qualifier");
1241659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis    if (RQ == CXRefQualifier_RValue)
1242659837e0ce0f73c7fdd5941854be3500db2f4013Argyrios Kyrtzidis      printf(" rvalue-ref-qualifier");
1243e1403d2dcb06fc306e186c787a7bd0ca30e06edeBenjamin Kramer    /* Print the canonical type if it is different. */
124404c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    {
124510f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXType CT = clang_getCanonicalType(T);
124604c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      if (!clang_equalTypes(T, CT)) {
1247ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko        PrintTypeAndTypeKind(CT, " [canonicaltype=%s] [canonicaltypekind=%s]");
124804c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      }
124904c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    }
1250e1403d2dcb06fc306e186c787a7bd0ca30e06edeBenjamin Kramer    /* Print the return type if it exists. */
125104c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    {
125210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXType RT = clang_getCursorResultType(cursor);
125304c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      if (RT.kind != CXType_Invalid) {
1254ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko        PrintTypeAndTypeKind(RT, " [resulttype=%s] [resulttypekind=%s]");
125504c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      }
125604c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    }
1257d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    /* Print the argument types if they exist. */
1258d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    {
1259651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      int NumArgs = clang_Cursor_getNumArguments(cursor);
1260651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (NumArgs != -1 && NumArgs != 0) {
126147f1165c92bc4104e314223ed9ad251e914687c1Argyrios Kyrtzidis        int i;
1262d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        printf(" [args=");
1263651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        for (i = 0; i < NumArgs; ++i) {
126410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella          CXType T = clang_getCursorType(clang_Cursor_getArgument(cursor, i));
1265d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          if (T.kind != CXType_Invalid) {
1266ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko            PrintTypeAndTypeKind(T, " [%s] [%s]");
1267d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          }
1268d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        }
1269d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        printf("]");
1270d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis      }
1271d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    }
1272651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    /* Print the template argument types if they exist. */
1273651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    {
1274651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      int NumTArgs = clang_Type_getNumTemplateArguments(T);
1275651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (NumTArgs != -1 && NumTArgs != 0) {
1276651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        int i;
1277651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        printf(" [templateargs/%d=", NumTArgs);
1278651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        for (i = 0; i < NumTArgs; ++i) {
1279651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          CXType TArg = clang_Type_getTemplateArgumentAsType(T, i);
1280651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          if (TArg.kind != CXType_Invalid) {
1281651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines            PrintTypeAndTypeKind(TArg, " [type=%s] [typekind=%s]");
1282651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          }
1283651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        }
1284651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        printf("]");
1285651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      }
1286651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    }
12873ce9e7d270e7df86c09c8126b4412d55be7c123bTed Kremenek    /* Print if this is a non-POD type. */
12883ce9e7d270e7df86c09c8126b4412d55be7c123bTed Kremenek    printf(" [isPOD=%d]", clang_isPODType(T));
12896bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    /* Print the pointee type. */
12906bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    {
12916bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      CXType PT = clang_getPointeeType(T);
12926bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      if (PT.kind != CXType_Invalid) {
12936bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        PrintTypeAndTypeKind(PT, " [pointeetype=%s] [pointeekind=%s]");
12946bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      }
12956bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    }
129604c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek
12978e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    printf("\n");
12988e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  }
12998e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  return CXChildVisit_Recurse;
13008e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek}
13018e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
1302411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidisstatic enum CXChildVisitResult PrintTypeSize(CXCursor cursor, CXCursor p,
1303411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis                                             CXClientData d) {
1304411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  CXType T;
1305411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  enum CXCursorKind K = clang_getCursorKind(cursor);
1306411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  if (clang_isInvalid(K))
1307411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    return CXChildVisit_Recurse;
1308411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  T = clang_getCursorType(cursor);
1309411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  PrintCursor(cursor, NULL);
1310411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  PrintTypeAndTypeKind(T, " [type=%s] [typekind=%s]");
1311411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  /* Print the type sizeof if applicable. */
1312411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  {
1313411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    long long Size = clang_Type_getSizeOf(T);
1314411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    if (Size >= 0 || Size < -1 ) {
1315411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      printf(" [sizeof=%lld]", Size);
1316411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    }
1317411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  }
1318411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  /* Print the type alignof if applicable. */
1319411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  {
1320411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    long long Align = clang_Type_getAlignOf(T);
1321411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    if (Align >= 0 || Align < -1) {
1322411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      printf(" [alignof=%lld]", Align);
1323411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    }
1324411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  }
1325411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  /* Print the record field offset if applicable. */
1326411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  {
13276bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    CXString FieldSpelling = clang_getCursorSpelling(cursor);
13286bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    const char *FieldName = clang_getCString(FieldSpelling);
1329411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    /* recurse to get the root anonymous record parent */
1330411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    CXCursor Parent, Root;
13316bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    if (clang_getCursorKind(cursor) == CXCursor_FieldDecl) {
13326bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      CXString RootParentSpelling;
13336bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      const char *RootParentName = 0;
1334651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Parent = p;
1335411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      do {
13366bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        if (RootParentName != 0)
13376bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines          clang_disposeString(RootParentSpelling);
13386bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
1339411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis        Root = Parent;
13406bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        RootParentSpelling = clang_getCursorSpelling(Root);
13416bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        RootParentName = clang_getCString(RootParentSpelling);
1342411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis        Parent = clang_getCursorSemanticParent(Root);
13436bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      } while (clang_getCursorType(Parent).kind == CXType_Record &&
13446bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines               !strcmp(RootParentName, ""));
13456bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      clang_disposeString(RootParentSpelling);
1346411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      /* if RootParentName is "", record is anonymous. */
1347411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      {
1348411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis        long long Offset = clang_Type_getOffsetOf(clang_getCursorType(Root),
1349411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis                                                  FieldName);
1350411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis        printf(" [offsetof=%lld]", Offset);
1351411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      }
1352411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    }
13536bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    clang_disposeString(FieldSpelling);
1354411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  }
1355411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  /* Print if its a bitfield */
1356411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  {
1357411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    int IsBitfield = clang_Cursor_isBitField(cursor);
1358411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    if (IsBitfield)
1359411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis      printf(" [BitFieldSize=%d]", clang_getFieldDeclBitWidth(cursor));
1360411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  }
1361411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  printf("\n");
1362411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  return CXChildVisit_Recurse;
1363411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis}
1364411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis
13651eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko/******************************************************************************/
13661eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko/* Bitwidth testing.                                                          */
13671eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko/******************************************************************************/
13681eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko
13691eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenkostatic enum CXChildVisitResult PrintBitWidth(CXCursor cursor, CXCursor p,
13701eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko                                             CXClientData d) {
137102c1b863440b067f3d674bde237146ef54c17875NAKAMURA Takumi  int Bitwidth;
13721eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  if (clang_getCursorKind(cursor) != CXCursor_FieldDecl)
13731eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    return CXChildVisit_Recurse;
13741eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko
137502c1b863440b067f3d674bde237146ef54c17875NAKAMURA Takumi  Bitwidth = clang_getFieldDeclBitWidth(cursor);
13761eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  if (Bitwidth >= 0) {
13771eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    PrintCursor(cursor, NULL);
13781eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    printf(" bitwidth=%d\n", Bitwidth);
13791eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  }
13801eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko
13811eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  return CXChildVisit_Recurse;
13821eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko}
13838e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
13848e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/******************************************************************************/
13857d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/* Loading ASTs/source.                                                       */
13867d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
13877d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
1388625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbarstatic int perform_test_load(CXIndex Idx, CXTranslationUnit TU,
138998271567b6f10be85bcc75d0bbb67416c8cfb83aTed Kremenek                             const char *filter, const char *prefix,
1390ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             CXCursorVisitor Visitor,
1391f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             PostVisitTU PV,
1392f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             const char *CommentSchemaFile) {
1393e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1394fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek  if (prefix)
1395e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    FileCheckPrefix = prefix;
1396e3ee02a324559829142f387f75daf3da118442beTed Kremenek
1397e3ee02a324559829142f387f75daf3da118442beTed Kremenek  if (Visitor) {
1398e3ee02a324559829142f387f75daf3da118442beTed Kremenek    enum CXCursorKind K = CXCursor_NotImplemented;
1399e3ee02a324559829142f387f75daf3da118442beTed Kremenek    enum CXCursorKind *ck = &K;
1400e3ee02a324559829142f387f75daf3da118442beTed Kremenek    VisitorData Data;
1401e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1402e3ee02a324559829142f387f75daf3da118442beTed Kremenek    /* Perform some simple filtering. */
1403e3ee02a324559829142f387f75daf3da118442beTed Kremenek    if (!strcmp(filter, "all") || !strcmp(filter, "local")) ck = NULL;
1404358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    else if (!strcmp(filter, "all-display") ||
1405358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor             !strcmp(filter, "local-display")) {
1406358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      ck = NULL;
1407358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      want_display_name = 1;
1408358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
1409b1ffee6e574d57b79ce2defd1b57052649221c03Daniel Dunbar    else if (!strcmp(filter, "none")) K = (enum CXCursorKind) ~0;
1410e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "category")) K = CXCursor_ObjCCategoryDecl;
1411e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "interface")) K = CXCursor_ObjCInterfaceDecl;
1412e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "protocol")) K = CXCursor_ObjCProtocolDecl;
1413e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "function")) K = CXCursor_FunctionDecl;
1414e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "typedef")) K = CXCursor_TypedefDecl;
1415e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "scan-function")) Visitor = FunctionScanVisitor;
1416e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else {
1417e3ee02a324559829142f387f75daf3da118442beTed Kremenek      fprintf(stderr, "Unknown filter for -test-load-tu: %s\n", filter);
1418e3ee02a324559829142f387f75daf3da118442beTed Kremenek      return 1;
1419e3ee02a324559829142f387f75daf3da118442beTed Kremenek    }
1420e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1421e3ee02a324559829142f387f75daf3da118442beTed Kremenek    Data.TU = TU;
1422e3ee02a324559829142f387f75daf3da118442beTed Kremenek    Data.Filter = ck;
14236bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    Data.CommentSchemaFile = CommentSchemaFile;
1424e3ee02a324559829142f387f75daf3da118442beTed Kremenek    clang_visitChildren(clang_getTranslationUnitCursor(TU), Visitor, &Data);
1425e3ee02a324559829142f387f75daf3da118442beTed Kremenek  }
1426e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1427ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek  if (PV)
1428ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek    PV(TU);
1429e3ee02a324559829142f387f75daf3da118442beTed Kremenek
1430a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
143116ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
143216ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis    clang_disposeTranslationUnit(TU);
143316ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis    return -1;
143416ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis  }
143516ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis
14360d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek  clang_disposeTranslationUnit(TU);
14370d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek  return 0;
14380d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek}
14390d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
1440fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenekint perform_test_load_tu(const char *file, const char *filter,
1441ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                         const char *prefix, CXCursorVisitor Visitor,
1442ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                         PostVisitTU PV) {
1443625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXIndex Idx;
1444625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXTranslationUnit TU;
1445020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  int result;
1446e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  Idx = clang_createIndex(/* excludeDeclsFromPCH */
14470a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                          !strcmp(filter, "local") ? 1 : 0,
1448fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                          /* displayDiagnostics=*/1);
1449e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1450020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  if (!CreateTranslationUnit(Idx, file, &TU)) {
1451020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
1452625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    return 1;
1453020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  }
1454625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1455f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, prefix, Visitor, PV, NULL);
1456020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  clang_disposeIndex(Idx);
1457020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  return result;
1458625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar}
1459625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1460ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenekint perform_test_load_source(int argc, const char **argv,
1461ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             const char *filter, CXCursorVisitor Visitor,
1462ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             PostVisitTU PV) {
1463ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  CXIndex Idx;
1464ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  CXTranslationUnit TU;
1465f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentSchemaFile;
14664db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
14674db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int num_unsaved_files = 0;
1468651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
14694db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int result;
1470651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
1471ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  Idx = clang_createIndex(/* excludeDeclsFromPCH */
1472358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                          (!strcmp(filter, "local") ||
1473358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                           !strcmp(filter, "local-display"))? 1 : 0,
1474cd6dcb3434d77b1f652416f2b5c3c62394f1472cArgyrios Kyrtzidis                          /* displayDiagnostics=*/1);
1475ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1476f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if ((CommentSchemaFile = parse_comments_schema(argc, argv))) {
1477f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    argc--;
1478f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    argv++;
1479f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
1480f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
1481020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
1482020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
14834db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    return -1;
1484020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  }
14854db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
1486651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(Idx, 0,
1487651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + num_unsaved_files,
1488651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - num_unsaved_files,
1489651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    unsaved_files, num_unsaved_files,
1490651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    getDefaultParsingOptions(), &TU);
1491651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
1492ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar    fprintf(stderr, "Unable to load translation unit!\n");
1493651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
1494abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
1495020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
1496ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar    return 1;
1497ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  }
1498ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1499f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, NULL, Visitor, PV,
1500f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             CommentSchemaFile);
15014db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1502020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  clang_disposeIndex(Idx);
15034db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  return result;
1504ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar}
1505ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1506abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregorint perform_test_reparse_source(int argc, const char **argv, int trials,
1507abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                const char *filter, CXCursorVisitor Visitor,
1508abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                PostVisitTU PV) {
1509abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  CXIndex Idx;
1510abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  CXTranslationUnit TU;
1511abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
1512abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int num_unsaved_files = 0;
1513651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int compiler_arg_idx = 0;
1514651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
1515651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  int result, i;
1516abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int trial;
151740098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  int remap_after_trial = 0;
151840098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  char *endptr = 0;
1519abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1520abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  Idx = clang_createIndex(/* excludeDeclsFromPCH */
1521abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                          !strcmp(filter, "local") ? 1 : 0,
1522cd6dcb3434d77b1f652416f2b5c3c62394f1472cArgyrios Kyrtzidis                          /* displayDiagnostics=*/1);
1523abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1524abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
1525abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    clang_disposeIndex(Idx);
1526abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    return -1;
1527abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1528651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
1529651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  for (i = 0; i < argc; ++i) {
1530651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (strcmp(argv[i], "--") == 0)
1531651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      break;
1532651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  }
1533651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (i < argc)
1534651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    compiler_arg_idx = i+1;
1535651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (num_unsaved_files > compiler_arg_idx)
1536651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    compiler_arg_idx = num_unsaved_files;
1537abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1538c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar  /* Load the initial translation unit -- we do this without honoring remapped
1539c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar   * files, so that we have a way to test results after changing the source. */
1540651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(Idx, 0,
1541651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + compiler_arg_idx,
1542651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - compiler_arg_idx,
1543651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    0, 0, getDefaultParsingOptions(), &TU);
1544651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
1545abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
1546651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
1547abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
1548abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    clang_disposeIndex(Idx);
1549abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    return 1;
1550abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1551abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1552bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
1553bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis    return -1;
1554bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
155540098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  if (getenv("CINDEXTEST_REMAP_AFTER_TRIAL")) {
155640098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis    remap_after_trial =
155740098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis        strtol(getenv("CINDEXTEST_REMAP_AFTER_TRIAL"), &endptr, 10);
155840098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  }
155940098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis
1560abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  for (trial = 0; trial < trials; ++trial) {
1561651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    free_remapped_files(unsaved_files, num_unsaved_files);
1562651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (parse_remapped_files_with_try(trial, argc, argv, 0,
1563651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                      &unsaved_files, &num_unsaved_files)) {
1564651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      clang_disposeTranslationUnit(TU);
1565651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      clang_disposeIndex(Idx);
1566651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      return -1;
1567651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    }
1568651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
1569651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    Err = clang_reparseTranslationUnit(
1570651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        TU,
1571651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        trial >= remap_after_trial ? num_unsaved_files : 0,
1572651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        trial >= remap_after_trial ? unsaved_files : 0,
1573651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        clang_defaultReparseOptions(TU));
1574651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (Err != CXError_Success) {
1575c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar      fprintf(stderr, "Unable to reparse translation unit!\n");
1576651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      describeLibclangFailure(Err);
1577abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      clang_disposeTranslationUnit(TU);
1578abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      free_remapped_files(unsaved_files, num_unsaved_files);
1579abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      clang_disposeIndex(Idx);
1580abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      return -1;
1581abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    }
1582bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
1583bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
1584bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis      return -1;
1585abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1586abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1587f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, NULL, Visitor, PV, NULL);
1588bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
1589abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1590abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  clang_disposeIndex(Idx);
1591abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  return result;
1592abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor}
1593abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
15940d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
15951c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/* Logic for testing clang_getCursor().                                       */
15961c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/******************************************************************************/
15971c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
1598dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregorstatic void print_cursor_file_scan(CXTranslationUnit TU, CXCursor cursor,
15991c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek                                   unsigned start_line, unsigned start_col,
16001d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                                   unsigned end_line, unsigned end_col,
16011d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                                   const char *prefix) {
16029096a20fc0fd799911b738ce7a9134dbacaf64b7Ted Kremenek  printf("// %s: ", FileCheckPrefix);
16031d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek  if (prefix)
16041d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    printf("-%s", prefix);
160551b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  PrintExtent(stdout, start_line, start_col, end_line, end_col);
160651b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  printf(" ");
1607f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(cursor, NULL);
16081c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  printf("\n");
16091c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
16101c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
16111d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenekstatic int perform_file_scan(const char *ast_file, const char *source_file,
16121d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                             const char *prefix) {
16131c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  CXIndex Idx;
16141c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  CXTranslationUnit TU;
16151c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  FILE *fp;
161610f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXCursor prevCursor = clang_getNullCursor();
1617b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  CXFile file;
16182389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  unsigned line = 1, col = 1;
16198f0bf81c0f3ab31881cc7db7914e6978bb2a19afDaniel Dunbar  unsigned start_line = 1, start_col = 1;
1620e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
16210a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
1622fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                                /* displayDiagnostics=*/1))) {
16231c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Could not create Index\n");
16241c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
16251c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1626e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
16271c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  if (!CreateTranslationUnit(Idx, ast_file, &TU))
16281c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
1629e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
16301c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  if ((fp = fopen(source_file, "r")) == NULL) {
16311c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Could not open '%s'\n", source_file);
16321c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
16331c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1634e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1635b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  file = clang_getFile(TU, source_file);
16362389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  for (;;) {
16372389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    CXCursor cursor;
16382389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    int c = fgetc(fp);
16392389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar
16402389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if (c == '\n') {
16412389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      ++line;
16422389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      col = 1;
16432389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    } else
16442389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      ++col;
16452389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar
16462389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    /* Check the cursor at this position, and dump the previous one if we have
16472389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar     * found something new.
16482389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar     */
16492389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    cursor = clang_getCursor(TU, clang_getLocation(TU, file, line, col));
16502389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if ((c == EOF || !clang_equalCursors(cursor, prevCursor)) &&
16512389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar        prevCursor.kind != CXCursor_InvalidFile) {
1652dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor      print_cursor_file_scan(TU, prevCursor, start_line, start_col,
1653d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar                             line, col, prefix);
16542389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      start_line = line;
16552389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      start_col = col;
1656a9933b98399f656653a0876fc39e5b9093efb732Benjamin Kramer    }
16572389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if (c == EOF)
16582389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      break;
1659a9933b98399f656653a0876fc39e5b9093efb732Benjamin Kramer
16602389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    prevCursor = cursor;
16611c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1662e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
16631c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  fclose(fp);
16644f5e21e24fb9e6ec473a13f83b5c9a2c41501a70Douglas Gregor  clang_disposeTranslationUnit(TU);
16654f5e21e24fb9e6ec473a13f83b5c9a2c41501a70Douglas Gregor  clang_disposeIndex(Idx);
16661c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  return 0;
16671c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
16681c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
16691c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/******************************************************************************/
167032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor/* Logic for testing clang code completion.                                   */
16710d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
16720d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
16730c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor/* Parse file:line:column from the input string. Returns 0 on success, non-zero
16740c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor   on failure. If successful, the pointer *filename will contain newly-allocated
16750c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor   memory (that will be owned by the caller) to store the file name. */
1676e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekint parse_file_line_column(const char *input, char **filename, unsigned *line,
1677fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                           unsigned *column, unsigned *second_line,
1678fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                           unsigned *second_column) {
167988d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor  /* Find the second colon. */
1680fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  const char *last_colon = strrchr(input, ':');
1681fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned values[4], i;
1682fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned num_values = (second_line && second_column)? 4 : 2;
1683fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
16840c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  char *endptr = 0;
1685fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!last_colon || last_colon == input) {
1686fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (num_values == 4)
1687fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      fprintf(stderr, "could not parse filename:line:column:line:column in "
1688fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              "'%s'\n", input);
1689fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    else
1690fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      fprintf(stderr, "could not parse filename:line:column in '%s'\n", input);
16910c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor    return 1;
16920c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
16930c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1694fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  for (i = 0; i != num_values; ++i) {
1695fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *prev_colon;
169688d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor
1697fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    /* Parse the next line or column. */
1698fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    values[num_values - i - 1] = strtol(last_colon + 1, &endptr, 10);
1699fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (*endptr != 0 && *endptr != ':') {
1700e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr, "could not parse %s in '%s'\n",
1701fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              (i % 2 ? "column" : "line"), input);
1702fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      return 1;
1703fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
1704e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1705fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (i + 1 == num_values)
1706fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      break;
170788d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor
1708fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    /* Find the previous colon. */
1709fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    prev_colon = last_colon - 1;
1710fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    while (prev_colon != input && *prev_colon != ':')
1711fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      --prev_colon;
1712fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (prev_colon == input) {
1713e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr, "could not parse %s in '%s'\n",
1714fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              (i % 2 == 0? "column" : "line"), input);
1715e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return 1;
1716fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
1717fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
1718fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    last_colon = prev_colon;
17190c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
1720fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
1721fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *line = values[0];
1722fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *column = values[1];
1723e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1724fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (second_line && second_column) {
1725fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *second_line = values[2];
1726fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *second_column = values[3];
1727fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
1728fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
172988d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor  /* Copy the file name. */
1730fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *filename = (char*)malloc(last_colon - input + 1);
1731fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  memcpy(*filename, input, last_colon - input);
1732fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  (*filename)[last_colon - input] = 0;
17330c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  return 0;
17340c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
17350c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
17360c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregorconst char *
17370c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregorclang_getCompletionChunkKindSpelling(enum CXCompletionChunkKind Kind) {
17380c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  switch (Kind) {
17390c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Optional: return "Optional";
17400c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_TypedText: return "TypedText";
17410c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Text: return "Text";
17420c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Placeholder: return "Placeholder";
17430c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Informative: return "Informative";
17440c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_CurrentParameter: return "CurrentParameter";
17450c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftParen: return "LeftParen";
17460c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightParen: return "RightParen";
17470c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftBracket: return "LeftBracket";
17480c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightBracket: return "RightBracket";
17490c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftBrace: return "LeftBrace";
17500c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightBrace: return "RightBrace";
17510c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftAngle: return "LeftAngle";
17520c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightAngle: return "RightAngle";
17530c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Comma: return "Comma";
1754ff5ce6eefc7c253ef6edf4d4bfc996fdd82d09aaDouglas Gregor  case CXCompletionChunk_ResultType: return "ResultType";
175501dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_Colon: return "Colon";
175601dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_SemiColon: return "SemiColon";
175701dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_Equal: return "Equal";
175801dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_HorizontalSpace: return "HorizontalSpace";
175901dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_VerticalSpace: return "VerticalSpace";
17600c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
1761e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
17620c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  return "Unknown";
17630c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
17640c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1765dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidisstatic int checkForErrors(CXTranslationUnit TU) {
1766dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  unsigned Num, i;
1767dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  CXDiagnostic Diag;
1768dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  CXString DiagStr;
1769dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1770dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (!getenv("CINDEXTEST_FAILONERROR"))
1771dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return 0;
1772dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1773dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  Num = clang_getNumDiagnostics(TU);
1774dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  for (i = 0; i != Num; ++i) {
1775dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    Diag = clang_getDiagnostic(TU, i);
1776dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (clang_getDiagnosticSeverity(Diag) >= CXDiagnostic_Error) {
1777dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      DiagStr = clang_formatDiagnostic(Diag,
1778dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis                                       clang_defaultDiagnosticDisplayOptions());
1779dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      fprintf(stderr, "%s\n", clang_getCString(DiagStr));
1780dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      clang_disposeString(DiagStr);
1781dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      clang_disposeDiagnostic(Diag);
1782dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
1783dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    }
1784dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    clang_disposeDiagnostic(Diag);
1785dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
1786dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1787dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  return 0;
1788dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis}
1789dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
17906bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic void print_completion_string(CXCompletionString completion_string,
17916bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                    FILE *file) {
1792f8297f1512d29c88f20bf2901f04cc04182a3eeeDaniel Dunbar  int I, N;
1793e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
17943ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  N = clang_getNumCompletionChunks(completion_string);
17950c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  for (I = 0; I != N; ++I) {
17962ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    CXString text;
17972ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    const char *cstr;
17980c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor    enum CXCompletionChunkKind Kind
17993ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      = clang_getCompletionChunkKind(completion_string, I);
1800e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
18013ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor    if (Kind == CXCompletionChunk_Optional) {
18023ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      fprintf(file, "{Optional ");
18033ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      print_completion_string(
1804e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                clang_getCompletionChunkCompletionString(completion_string, I),
18053ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor                              file);
18063ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      fprintf(file, "}");
18073ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      continue;
18085a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor    }
18095a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor
18105a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor    if (Kind == CXCompletionChunk_VerticalSpace) {
18115a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor      fprintf(file, "{VerticalSpace  }");
18125a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor      continue;
18133ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor    }
1814e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1815d5a2089663d81faee0de031deabc418fa53ecf3bDouglas Gregor    text = clang_getCompletionChunkText(completion_string, I);
18162ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    cstr = clang_getCString(text);
1817e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(file, "{%s %s}",
18180c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor            clang_getCompletionChunkKindSpelling(Kind),
18192ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek            cstr ? cstr : "");
18202ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    clang_disposeString(text);
18210c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
18222ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek
18233ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor}
18243ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor
18256bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic void print_completion_result(CXCompletionResult *completion_result,
18266bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                    FILE *file) {
182710f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXString ks = clang_getCursorKindSpelling(completion_result->CursorKind);
18286164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  unsigned annotationCount;
1829ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  enum CXCursorKind ParentKind;
1830ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  CXString ParentName;
1831d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  CXString BriefComment;
1832d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  const char *BriefCommentCString;
1833ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor
1834e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  fprintf(file, "%s:", clang_getCString(ks));
1835e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  clang_disposeString(ks);
1836e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
18373ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  print_completion_string(completion_result->CompletionString, file);
183858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  fprintf(file, " (%u)",
183912e131385e892e3723483a1081a89bcad29c8a84Douglas Gregor          clang_getCompletionPriority(completion_result->CompletionString));
184058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  switch (clang_getCompletionAvailability(completion_result->CompletionString)){
184158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_Available:
184258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
184358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
184458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_Deprecated:
184558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    fprintf(file, " (deprecated)");
184658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
184758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
184858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_NotAvailable:
184958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    fprintf(file, " (unavailable)");
185058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
1851d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen
1852d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen  case CXAvailability_NotAccessible:
1853d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen    fprintf(file, " (inaccessible)");
1854d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen    break;
185558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  }
18566164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen
18576164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  annotationCount = clang_getCompletionNumAnnotations(
18586164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen        completion_result->CompletionString);
18596164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  if (annotationCount) {
18606164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    unsigned i;
18616164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    fprintf(file, " (");
18626164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    for (i = 0; i < annotationCount; ++i) {
18636164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen      if (i != 0)
18646164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen        fprintf(file, ", ");
18656164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen      fprintf(file, "\"%s\"",
18666164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen              clang_getCString(clang_getCompletionAnnotation(
18676164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen                                 completion_result->CompletionString, i)));
18686164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    }
18696164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    fprintf(file, ")");
18706164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  }
18716164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen
1872ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  if (!getenv("CINDEXTEST_NO_COMPLETION_PARENTS")) {
1873ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    ParentName = clang_getCompletionParent(completion_result->CompletionString,
1874ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor                                           &ParentKind);
1875ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    if (ParentKind != CXCursor_NotImplemented) {
187610f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella      CXString KindSpelling = clang_getCursorKindSpelling(ParentKind);
1877ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      fprintf(file, " (parent: %s '%s')",
1878ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor              clang_getCString(KindSpelling),
1879ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor              clang_getCString(ParentName));
1880ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      clang_disposeString(KindSpelling);
1881ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    }
1882ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    clang_disposeString(ParentName);
1883ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  }
1884d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko
1885d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  BriefComment = clang_getCompletionBriefComment(
1886d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko                                        completion_result->CompletionString);
1887d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  BriefCommentCString = clang_getCString(BriefComment);
1888d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (BriefCommentCString && *BriefCommentCString != '\0') {
1889d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    fprintf(file, "(brief comment: %s)", BriefCommentCString);
1890d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  }
1891d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  clang_disposeString(BriefComment);
1892ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor
189358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  fprintf(file, "\n");
18940c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
18950c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
18963da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregorvoid print_completion_contexts(unsigned long long contexts, FILE *file) {
18973da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  fprintf(file, "Completion contexts:\n");
18983da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts == CXCompletionContext_Unknown) {
18993da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Unknown\n");
19003da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19013da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_AnyType) {
19023da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Any type\n");
19033da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19043da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_AnyValue) {
19053da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Any value\n");
19063da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19073da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCObjectValue) {
19083da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C object value\n");
19093da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19103da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCSelectorValue) {
19113da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C selector value\n");
19123da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19133da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_CXXClassTypeValue) {
19143da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "C++ class type value\n");
19153da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19163da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_DotMemberAccess) {
19173da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Dot member access\n");
19183da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19193da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ArrowMemberAccess) {
19203da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Arrow member access\n");
19213da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19223da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCPropertyAccess) {
19233da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C property access\n");
19243da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19253da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_EnumTag) {
19263da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Enum tag\n");
19273da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19283da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_UnionTag) {
19293da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Union tag\n");
19303da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19313da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_StructTag) {
19323da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Struct tag\n");
19333da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19343da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ClassTag) {
19353da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Class name\n");
19363da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19373da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_Namespace) {
19383da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Namespace or namespace alias\n");
19393da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19403da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_NestedNameSpecifier) {
19413da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Nested name specifier\n");
19423da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19433da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCInterface) {
19443da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C interface\n");
19453da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19463da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCProtocol) {
19473da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C protocol\n");
19483da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19493da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCCategory) {
19503da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C category\n");
19513da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19523da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCInstanceMessage) {
19533da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C instance method\n");
19543da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19553da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCClassMessage) {
19563da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C class method\n");
19573da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19583da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCSelectorName) {
19593da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C selector name\n");
19603da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19613da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_MacroName) {
19623da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Macro name\n");
19633da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19643da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_NaturalLanguage) {
19653da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Natural language\n");
19663da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
19673da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor}
19683da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
19691e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregorint my_stricmp(const char *s1, const char *s2) {
19701e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  while (*s1 && *s2) {
19716d5552131a4637f9bbe8c93386648e9bbb2c30feNAKAMURA Takumi    int c1 = tolower((unsigned char)*s1), c2 = tolower((unsigned char)*s2);
19721e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    if (c1 < c2)
19731e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      return -1;
19741e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    else if (c1 > c2)
19751e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      return 1;
19761e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
19771e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    ++s1;
19781e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    ++s2;
19791e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  }
19801e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
19811e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  if (*s1)
19821e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    return 1;
19831e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  else if (*s2)
19841e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    return -1;
19851e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  return 0;
19861e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor}
19871e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
19881982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregorint perform_code_completion(int argc, const char **argv, int timing_only) {
19890c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  const char *input = argv[1];
19900c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  char *filename = 0;
19910c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  unsigned line;
19920c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  unsigned column;
1993f8297f1512d29c88f20bf2901f04cc04182a3eeeDaniel Dunbar  CXIndex CIdx;
1994f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  int errorCode;
1995735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
1996735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  int num_unsaved_files = 0;
1997ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  CXCodeCompleteResults *results = 0;
1998651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
1999651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  CXTranslationUnit TU;
200032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  unsigned I, Repeats = 1;
200132be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  unsigned completionOptions = clang_defaultCodeCompleteOptions();
200232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
200332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (getenv("CINDEXTEST_CODE_COMPLETE_PATTERNS"))
200432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    completionOptions |= CXCodeComplete_IncludeCodePatterns;
2005d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (getenv("CINDEXTEST_COMPLETION_BRIEF_COMMENTS"))
2006d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    completionOptions |= CXCodeComplete_IncludeBriefComments;
2007df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor
20081982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  if (timing_only)
20091982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    input += strlen("-code-completion-timing=");
20101982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  else
20111982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    input += strlen("-code-completion-at=");
20121982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor
2013e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if ((errorCode = parse_file_line_column(input, &filename, &line, &column,
2014fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                          0, 0)))
2015f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek    return errorCode;
20160c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
2017735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  if (parse_remapped_files(argc, argv, 2, &unsaved_files, &num_unsaved_files))
2018735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor    return -1;
2019735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor
202032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  CIdx = clang_createIndex(0, 0);
202132be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
202232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (getenv("CINDEXTEST_EDITING"))
202332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    Repeats = 5;
2024651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
2025651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(CIdx, 0,
2026651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + num_unsaved_files + 2,
2027651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - num_unsaved_files - 2,
2028651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    0, 0, getDefaultParsingOptions(), &TU);
2029651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
203032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
2031651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
203232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    return 1;
203332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  }
203408bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor
2035651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_reparseTranslationUnit(TU, 0, 0,
2036651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                     clang_defaultReparseOptions(TU));
2037651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
2038651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
203908bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor    fprintf(stderr, "Unable to reparse translation init!\n");
2040651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
2041651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_disposeTranslationUnit(TU);
204208bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor    return 1;
204308bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor  }
2044651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
204532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  for (I = 0; I != Repeats; ++I) {
204632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    results = clang_codeCompleteAt(TU, filename, line, column,
204732be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                   unsaved_files, num_unsaved_files,
204832be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                   completionOptions);
204932be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    if (!results) {
205032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor      fprintf(stderr, "Unable to perform code completion!\n");
20512de41c9c99e0e9ae6488d04c08423a5c1190109eDaniel Dunbar      return 1;
20522de41c9c99e0e9ae6488d04c08423a5c1190109eDaniel Dunbar    }
205332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    if (I != Repeats-1)
205432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor      clang_disposeCodeCompleteResults(results);
205532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  }
2056936ea3b590117d2cd73b1b92621d06c4a7edbe60Douglas Gregor
2057ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  if (results) {
2058e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    unsigned i, n = results->NumResults, containerIsIncomplete = 0;
20593da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    unsigned long long contexts;
2060e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    enum CXCursorKind containerKind;
20610a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    CXString objCSelector;
20620a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    const char *selectorString;
20631e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    if (!timing_only) {
20641e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      /* Sort the code-completion results based on the typed text. */
20651e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      clang_sortCodeCompletionResults(results->Results, results->NumResults);
20661e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
20671982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor      for (i = 0; i != n; ++i)
20681982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor        print_completion_result(results->Results + i, stdout);
20691e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    }
2070a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    n = clang_codeCompleteGetNumDiagnostics(results);
2071a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    for (i = 0; i != n; ++i) {
2072a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      CXDiagnostic diag = clang_codeCompleteGetDiagnostic(results, i);
2073a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      PrintDiagnostic(diag);
2074a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      clang_disposeDiagnostic(diag);
2075a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    }
20763da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
20773da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    contexts = clang_codeCompleteGetContexts(results);
20783da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    print_completion_contexts(contexts, stdout);
20793da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
20800a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    containerKind = clang_codeCompleteGetContainerKind(results,
20810a47d69af8bda945352997af3da4687a3356096aDouglas Gregor                                                       &containerIsIncomplete);
2082e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
2083e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    if (containerKind != CXCursor_InvalidCode) {
2084e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      /* We have found a container */
2085e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      CXString containerUSR, containerKindSpelling;
2086e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      containerKindSpelling = clang_getCursorKindSpelling(containerKind);
2087e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      printf("Container Kind: %s\n", clang_getCString(containerKindSpelling));
2088e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      clang_disposeString(containerKindSpelling);
2089e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
2090e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      if (containerIsIncomplete) {
2091e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor        printf("Container is incomplete\n");
2092e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      }
2093e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      else {
2094e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor        printf("Container is complete\n");
2095e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      }
2096e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
2097e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      containerUSR = clang_codeCompleteGetContainerUSR(results);
2098e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      printf("Container USR: %s\n", clang_getCString(containerUSR));
2099e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      clang_disposeString(containerUSR);
2100e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    }
2101e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
21020a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    objCSelector = clang_codeCompleteGetObjCSelector(results);
21030a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    selectorString = clang_getCString(objCSelector);
21040a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    if (selectorString && strlen(selectorString) > 0) {
21050a47d69af8bda945352997af3da4687a3356096aDouglas Gregor      printf("Objective-C selector: %s\n", selectorString);
21060a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    }
21070a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    clang_disposeString(objCSelector);
21080a47d69af8bda945352997af3da4687a3356096aDouglas Gregor
2109ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor    clang_disposeCodeCompleteResults(results);
2110ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  }
2111df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor  clang_disposeTranslationUnit(TU);
21120c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  clang_disposeIndex(CIdx);
21130c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  free(filename);
2114e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2115735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
2116735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor
2117f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  return 0;
21180c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
21190c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
2120f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregortypedef struct {
2121f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  char *filename;
2122f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned line;
2123f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned column;
2124f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor} CursorSourceLocation;
2125f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor
2126aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic int inspect_cursor_at(int argc, const char **argv) {
2127f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXIndex CIdx;
2128f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  int errorCode;
2129f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
2130f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  int num_unsaved_files = 0;
2131651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
2132f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXTranslationUnit TU;
2133f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXCursor Cursor;
2134f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CursorSourceLocation *Locations = 0;
2135f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned NumLocations = 0, Loc;
21368e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  unsigned Repeats = 1;
2137bdc4b366e80c125184a3b3c56fa4619cb4ac9e45Douglas Gregor  unsigned I;
21388e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
2139e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  /* Count the number of locations. */
2140f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  while (strstr(argv[NumLocations+1], "-cursor-at=") == argv[NumLocations+1])
2141f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    ++NumLocations;
2142e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2143f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  /* Parse the locations. */
2144f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  assert(NumLocations > 0 && "Unable to count locations?");
2145f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  Locations = (CursorSourceLocation *)malloc(
2146f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                                  NumLocations * sizeof(CursorSourceLocation));
2147f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  for (Loc = 0; Loc < NumLocations; ++Loc) {
2148f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    const char *input = argv[Loc + 1] + strlen("-cursor-at=");
2149e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    if ((errorCode = parse_file_line_column(input, &Locations[Loc].filename,
2150e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                                            &Locations[Loc].line,
2151fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                            &Locations[Loc].column, 0, 0)))
2152f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor      return errorCode;
2153f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
2154e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2155e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (parse_remapped_files(argc, argv, NumLocations + 1, &unsaved_files,
2156f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                           &num_unsaved_files))
2157f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return -1;
2158e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
21598e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  if (getenv("CINDEXTEST_EDITING"))
21608e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    Repeats = 5;
21618e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
21628e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  /* Parse the translation unit. When we're testing clang_getCursor() after
21638e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor     reparsing, don't remap unsaved files until the second parse. */
21648e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  CIdx = clang_createIndex(1, 1);
2165651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(CIdx, argv[argc - 1],
2166651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   argv + num_unsaved_files + 1 + NumLocations,
2167651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   argc - num_unsaved_files - 2 - NumLocations,
2168651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   unsaved_files,
2169651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   Repeats > 1? 0 : num_unsaved_files,
2170651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                   getDefaultParsingOptions(), &TU);
2171651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
2172f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    fprintf(stderr, "unable to parse input\n");
2173651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
2174f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return -1;
2175f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
2176e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
2177dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
2178dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return -1;
2179dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2180bdc4b366e80c125184a3b3c56fa4619cb4ac9e45Douglas Gregor  for (I = 0; I != Repeats; ++I) {
2181651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (Repeats > 1) {
2182651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Err = clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2183651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         clang_defaultReparseOptions(TU));
2184651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (Err != CXError_Success) {
2185651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        describeLibclangFailure(Err);
2186651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        clang_disposeTranslationUnit(TU);
2187651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        return 1;
2188651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      }
21898e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    }
2190dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2191dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
2192dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
21938e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
21948e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    for (Loc = 0; Loc < NumLocations; ++Loc) {
21958e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      CXFile file = clang_getFile(TU, Locations[Loc].filename);
21968e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      if (!file)
21978e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        continue;
21988e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
21998e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      Cursor = clang_getCursor(TU,
22008e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                               clang_getLocation(TU, file, Locations[Loc].line,
22018e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                                 Locations[Loc].column));
2202dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2203dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      if (checkForErrors(TU) != 0)
2204dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        return -1;
2205dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
22068e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      if (I + 1 == Repeats) {
220710f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        CXCompletionString completionString = clang_getCursorCompletionString(
220810f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella                                                                        Cursor);
220910f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella        CXSourceLocation CursorLoc = clang_getCursorLocation(Cursor);
221066373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        CXString Spelling;
221166373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        const char *cspell;
221266373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        unsigned line, column;
221366373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        clang_getSpellingLocation(CursorLoc, 0, &line, &column, 0);
221466373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        printf("%d:%d ", line, column);
2215f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCursor(Cursor, NULL);
221666373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        PrintCursorExtent(Cursor);
221766373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        Spelling = clang_getCursorSpelling(Cursor);
221866373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        cspell = clang_getCString(Spelling);
2219ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis        if (cspell && strlen(cspell) != 0) {
2220ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          unsigned pieceIndex;
2221ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          printf(" Spelling=%s (", cspell);
2222ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          for (pieceIndex = 0; ; ++pieceIndex) {
222310f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella            CXSourceRange range =
222410f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella              clang_Cursor_getSpellingNameRange(Cursor, pieceIndex, 0);
2225ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis            if (clang_Range_isNull(range))
2226ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis              break;
2227ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis            PrintRange(range, 0);
2228ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          }
2229ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          printf(")");
2230ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis        }
223166373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        clang_disposeString(Spelling);
223234ebe1e1b0779bcea2f277bc6b4e9dd98bf70b7bArgyrios Kyrtzidis        if (clang_Cursor_getObjCSelectorIndex(Cursor) != -1)
22336bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines          printf(" Selector index=%d",
22346bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                 clang_Cursor_getObjCSelectorIndex(Cursor));
2235f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis        if (clang_Cursor_isDynamicCall(Cursor))
2236f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis          printf(" Dynamic-call");
2237e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis        if (Cursor.kind == CXCursor_ObjCMessageExpr) {
223810f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella          CXType T = clang_Cursor_getReceiverType(Cursor);
223910f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella          CXString S = clang_getTypeKindSpelling(T.kind);
2240e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis          printf(" Receiver-type=%s", clang_getCString(S));
2241e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis          clang_disposeString(S);
2242e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis        }
2243f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis
22445d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis        {
22455d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis          CXModule mod = clang_Cursor_getModule(Cursor);
2246e858e667c14ce4a9df5a4bbae770a0a3a3c8723eArgyrios Kyrtzidis          CXFile astFile;
2247e858e667c14ce4a9df5a4bbae770a0a3a3c8723eArgyrios Kyrtzidis          CXString name, astFilename;
22485d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis          unsigned i, numHeaders;
22495d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis          if (mod) {
2250e858e667c14ce4a9df5a4bbae770a0a3a3c8723eArgyrios Kyrtzidis            astFile = clang_Module_getASTFile(mod);
2251e858e667c14ce4a9df5a4bbae770a0a3a3c8723eArgyrios Kyrtzidis            astFilename = clang_getFileName(astFile);
22525d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            name = clang_Module_getFullName(mod);
2253c1d22393628a145e54396c0ac66e9625d13a7658Argyrios Kyrtzidis            numHeaders = clang_Module_getNumTopLevelHeaders(TU, mod);
22546bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines            printf(" ModuleName=%s (%s) system=%d Headers(%d):",
2255e858e667c14ce4a9df5a4bbae770a0a3a3c8723eArgyrios Kyrtzidis                   clang_getCString(name), clang_getCString(astFilename),
22566bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                   clang_Module_isSystem(mod), numHeaders);
22575d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            clang_disposeString(name);
2258e858e667c14ce4a9df5a4bbae770a0a3a3c8723eArgyrios Kyrtzidis            clang_disposeString(astFilename);
22595d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            for (i = 0; i < numHeaders; ++i) {
226010f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella              CXFile file = clang_Module_getTopLevelHeader(TU, mod, i);
226110f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella              CXString filename = clang_getFileName(file);
22625d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis              printf("\n%s", clang_getCString(filename));
22635d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis              clang_disposeString(filename);
22645d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            }
22655d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis          }
22665d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis        }
22675d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis
22688fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor        if (completionString != NULL) {
22698fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor          printf("\nCompletion string: ");
22708fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor          print_completion_string(completionString, stdout);
22718fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor        }
22728e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        printf("\n");
22738e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        free(Locations[Loc].filename);
22748e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      }
22758e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    }
2276f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
22778e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
2278a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
2279f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  clang_disposeTranslationUnit(TU);
2280f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  clang_disposeIndex(CIdx);
2281f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  free(Locations);
2282f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
2283f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  return 0;
2284f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor}
2285f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor
2286aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic enum CXVisitorResult findFileRefsVisit(void *context,
2287aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                         CXCursor cursor, CXSourceRange range) {
2288aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (clang_Range_isNull(range))
2289aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return CXVisit_Continue;
2290aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2291f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(cursor, NULL);
2292aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintRange(range, "");
2293aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  printf("\n");
2294aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  return CXVisit_Continue;
2295aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis}
2296aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2297aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic int find_file_refs_at(int argc, const char **argv) {
2298aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXIndex CIdx;
2299aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  int errorCode;
2300aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  struct CXUnsavedFile *unsaved_files = 0;
2301aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  int num_unsaved_files = 0;
2302651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
2303aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXTranslationUnit TU;
2304aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXCursor Cursor;
2305aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CursorSourceLocation *Locations = 0;
2306aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned NumLocations = 0, Loc;
2307aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned Repeats = 1;
2308aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned I;
2309aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2310aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Count the number of locations. */
2311aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  while (strstr(argv[NumLocations+1], "-file-refs-at=") == argv[NumLocations+1])
2312aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    ++NumLocations;
2313aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2314aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Parse the locations. */
2315aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  assert(NumLocations > 0 && "Unable to count locations?");
2316aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  Locations = (CursorSourceLocation *)malloc(
2317aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  NumLocations * sizeof(CursorSourceLocation));
2318aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  for (Loc = 0; Loc < NumLocations; ++Loc) {
2319aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    const char *input = argv[Loc + 1] + strlen("-file-refs-at=");
2320aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if ((errorCode = parse_file_line_column(input, &Locations[Loc].filename,
2321aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                            &Locations[Loc].line,
2322aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                            &Locations[Loc].column, 0, 0)))
2323aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return errorCode;
2324aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2325aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2326aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (parse_remapped_files(argc, argv, NumLocations + 1, &unsaved_files,
2327aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                           &num_unsaved_files))
2328aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return -1;
2329aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2330aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING"))
2331aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    Repeats = 5;
2332aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2333aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Parse the translation unit. When we're testing clang_getCursor() after
2334aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis     reparsing, don't remap unsaved files until the second parse. */
2335aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CIdx = clang_createIndex(1, 1);
2336651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(CIdx, argv[argc - 1],
2337651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + num_unsaved_files + 1 + NumLocations,
2338651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - num_unsaved_files - 2 - NumLocations,
2339651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    unsaved_files,
2340651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    Repeats > 1? 0 : num_unsaved_files,
2341651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    getDefaultParsingOptions(), &TU);
2342651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
2343aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    fprintf(stderr, "unable to parse input\n");
2344651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
2345651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_disposeTranslationUnit(TU);
2346aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return -1;
2347aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2348aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2349dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
2350dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return -1;
2351dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2352aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  for (I = 0; I != Repeats; ++I) {
2353651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (Repeats > 1) {
2354651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Err = clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2355651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         clang_defaultReparseOptions(TU));
2356651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (Err != CXError_Success) {
2357651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        describeLibclangFailure(Err);
2358651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        clang_disposeTranslationUnit(TU);
2359651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        return 1;
2360651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      }
2361aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
2362dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2363dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
2364dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
2365aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2366aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    for (Loc = 0; Loc < NumLocations; ++Loc) {
2367aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      CXFile file = clang_getFile(TU, Locations[Loc].filename);
2368aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (!file)
2369aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        continue;
2370aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2371aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      Cursor = clang_getCursor(TU,
2372aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                               clang_getLocation(TU, file, Locations[Loc].line,
2373aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                                 Locations[Loc].column));
2374dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2375dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      if (checkForErrors(TU) != 0)
2376dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        return -1;
2377dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2378aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (I + 1 == Repeats) {
237926fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen        CXCursorAndRangeVisitor visitor = { 0, findFileRefsVisit };
2380f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCursor(Cursor, NULL);
2381aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        printf("\n");
2382aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        clang_findReferencesInFile(Cursor, file, visitor);
2383aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        free(Locations[Loc].filename);
2384dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2385dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        if (checkForErrors(TU) != 0)
2386dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis          return -1;
2387aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      }
2388aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
2389aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2390aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2391aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintDiagnostics(TU);
2392aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  clang_disposeTranslationUnit(TU);
2393aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  clang_disposeIndex(CIdx);
2394aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  free(Locations);
2395aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  free_remapped_files(unsaved_files, num_unsaved_files);
2396aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  return 0;
2397aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis}
2398aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2399ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidisstatic enum CXVisitorResult findFileIncludesVisit(void *context,
2400ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis                                         CXCursor cursor, CXSourceRange range) {
2401ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  PrintCursor(cursor, NULL);
2402ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  PrintRange(range, "");
2403ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  printf("\n");
2404ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  return CXVisit_Continue;
2405ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis}
2406ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2407ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidisstatic int find_file_includes_in(int argc, const char **argv) {
2408ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  CXIndex CIdx;
2409ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  struct CXUnsavedFile *unsaved_files = 0;
2410ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  int num_unsaved_files = 0;
2411651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
2412ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  CXTranslationUnit TU;
2413ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  const char **Filenames = 0;
2414ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  unsigned NumFilenames = 0;
2415ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  unsigned Repeats = 1;
2416ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  unsigned I, FI;
2417ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2418ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  /* Count the number of locations. */
2419ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  while (strstr(argv[NumFilenames+1], "-file-includes-in=") == argv[NumFilenames+1])
2420ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    ++NumFilenames;
2421ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2422ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  /* Parse the locations. */
2423ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  assert(NumFilenames > 0 && "Unable to count filenames?");
2424ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  Filenames = (const char **)malloc(NumFilenames * sizeof(const char *));
2425ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  for (I = 0; I < NumFilenames; ++I) {
2426ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    const char *input = argv[I + 1] + strlen("-file-includes-in=");
2427ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    /* Copy the file name. */
2428ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    Filenames[I] = input;
2429ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  }
2430ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2431ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  if (parse_remapped_files(argc, argv, NumFilenames + 1, &unsaved_files,
2432ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis                           &num_unsaved_files))
2433ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    return -1;
2434ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2435ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING"))
2436ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    Repeats = 2;
2437ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2438ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  /* Parse the translation unit. When we're testing clang_getCursor() after
2439ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis     reparsing, don't remap unsaved files until the second parse. */
2440ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  CIdx = clang_createIndex(1, 1);
2441651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(
2442651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      CIdx, argv[argc - 1],
2443651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      argv + num_unsaved_files + 1 + NumFilenames,
2444651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      argc - num_unsaved_files - 2 - NumFilenames,
2445651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      unsaved_files,
2446651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Repeats > 1 ? 0 : num_unsaved_files, getDefaultParsingOptions(), &TU);
2447651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
2448651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
2449ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    fprintf(stderr, "unable to parse input\n");
2450651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
2451651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_disposeTranslationUnit(TU);
2452ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    return -1;
2453ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  }
2454ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2455ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  if (checkForErrors(TU) != 0)
2456ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    return -1;
2457ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2458ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  for (I = 0; I != Repeats; ++I) {
2459651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    if (Repeats > 1) {
2460651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Err = clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2461651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         clang_defaultReparseOptions(TU));
2462651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (Err != CXError_Success) {
2463651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        describeLibclangFailure(Err);
2464651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        clang_disposeTranslationUnit(TU);
2465651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        return 1;
2466651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      }
2467ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    }
2468ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2469ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    if (checkForErrors(TU) != 0)
2470ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      return -1;
2471ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2472ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    for (FI = 0; FI < NumFilenames; ++FI) {
2473ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      CXFile file = clang_getFile(TU, Filenames[FI]);
2474ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      if (!file)
2475ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        continue;
2476ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2477ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      if (checkForErrors(TU) != 0)
2478ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        return -1;
2479ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2480ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      if (I + 1 == Repeats) {
2481ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        CXCursorAndRangeVisitor visitor = { 0, findFileIncludesVisit };
2482ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        clang_findIncludesInFile(TU, file, visitor);
2483ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2484ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis        if (checkForErrors(TU) != 0)
2485ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis          return -1;
2486ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis      }
2487ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    }
2488ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  }
2489ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
2490ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  PrintDiagnostics(TU);
2491ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  clang_disposeTranslationUnit(TU);
2492ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  clang_disposeIndex(CIdx);
24935256c1fd315104f7b1542bb231f16dbad8a8d64bArgyrios Kyrtzidis  free((void *)Filenames);
2494ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  free_remapped_files(unsaved_files, num_unsaved_files);
2495ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  return 0;
2496ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis}
2497ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis
249811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis#define MAX_IMPORTED_ASTFILES 200
249911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
250011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidistypedef struct {
250111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  char **filenames;
250211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  unsigned num_files;
250311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis} ImportedASTFilesData;
250411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
250511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic ImportedASTFilesData *importedASTs_create() {
250611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  ImportedASTFilesData *p;
250711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p = malloc(sizeof(ImportedASTFilesData));
250811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p->filenames = malloc(MAX_IMPORTED_ASTFILES * sizeof(const char *));
250911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p->num_files = 0;
251011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  return p;
251111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis}
251211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
251311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic void importedASTs_dispose(ImportedASTFilesData *p) {
251411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  unsigned i;
251511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (!p)
251611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    return;
251711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
251811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  for (i = 0; i < p->num_files; ++i)
251911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    free(p->filenames[i]);
252011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  free(p->filenames);
252111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  free(p);
252211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis}
252311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
252411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic void importedASTS_insert(ImportedASTFilesData *p, const char *file) {
252511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  unsigned i;
252611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  assert(p && file);
252711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  for (i = 0; i < p->num_files; ++i)
252811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    if (strcmp(file, p->filenames[i]) == 0)
252911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis      return;
253011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  assert(p->num_files + 1 < MAX_IMPORTED_ASTFILES);
253111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p->filenames[p->num_files++] = strdup(file);
253211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis}
253311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
25346bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinestypedef struct IndexDataStringList_ {
25356bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  struct IndexDataStringList_ *next;
25366bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  char data[1]; /* Dynamically sized. */
25376bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines} IndexDataStringList;
25386bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
25394e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidistypedef struct {
25404e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *check_prefix;
25414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  int first_check_printed;
2542dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  int fail_for_error;
25436f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  int abort;
254413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  const char *main_filename;
254511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  ImportedASTFilesData *importedASTs;
25466bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  IndexDataStringList *strings;
25476bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  CXTranslationUnit TU;
25484e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis} IndexData;
25494e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
25506bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic void free_client_data(IndexData *index_data) {
25516bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  IndexDataStringList *node = index_data->strings;
25526bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  while (node) {
25536bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    IndexDataStringList *next = node->next;
25546bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    free(node);
25556bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    node = next;
25566bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  }
25576bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data->strings = NULL;
25586bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines}
25596bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
25604e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void printCheck(IndexData *data) {
25614e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (data->check_prefix) {
25624e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    if (data->first_check_printed) {
25634e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      printf("// %s-NEXT: ", data->check_prefix);
25644e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    } else {
25654e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      printf("// %s     : ", data->check_prefix);
25664e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      data->first_check_printed = 1;
25674e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    }
25684e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
25694e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
25704e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2571dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic void printCXIndexFile(CXIdxClientFile file) {
257210f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella  CXString filename = clang_getFileName((CXFile)file);
25734e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("%s", clang_getCString(filename));
25744e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_disposeString(filename);
25754e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
25764e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
257713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidisstatic void printCXIndexLoc(CXIdxLoc loc, CXClientData client_data) {
257813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  IndexData *index_data;
25794e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString filename;
258013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  const char *cname;
2581dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  CXIdxClientFile file;
25824e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  unsigned line, column;
258313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  int isMainFile;
25844e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
258513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
25864e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_indexLoc_getFileLocation(loc, &file, 0, &line, &column, 0);
25874e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (line == 0) {
25888003fd613724847bba834ae41aca6f446af1f818Argyrios Kyrtzidis    printf("<invalid>");
25894e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    return;
25904e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
2591c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  if (!file) {
2592c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis    printf("<no idxfile>");
2593c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis    return;
2594c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  }
25954e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  filename = clang_getFileName((CXFile)file);
25964e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  cname = clang_getCString(filename);
259713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  if (strcmp(cname, index_data->main_filename) == 0)
259813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    isMainFile = 1;
259913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  else
260013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    isMainFile = 0;
260113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  clang_disposeString(filename);
260213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
260313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  if (!isMainFile) {
26044e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printCXIndexFile(file);
26054e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printf(":");
26064e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
26074e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("%d:%d", line, column);
26084e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
26094e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
261013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidisstatic unsigned digitCount(unsigned val) {
261113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  unsigned c = 1;
261213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  while (1) {
261313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    if (val < 10)
261413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis      return c;
261513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    ++c;
261613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    val /= 10;
261713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  }
261813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis}
261913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
26206bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic CXIdxClientContainer makeClientContainer(CXClientData *client_data,
26216bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                                const CXIdxEntityInfo *info,
26226ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                                CXIdxLoc loc) {
26236bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  IndexData *index_data;
26246bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  IndexDataStringList *node;
26254e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *name;
26264e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  char *newStr;
2627dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  CXIdxClientFile file;
26284e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  unsigned line, column;
26294e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2630dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  name = info->name;
26314e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (!name)
26324e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    name = "<anon-tag>";
26334e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
26344e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_indexLoc_getFileLocation(loc, &file, 0, &line, &column, 0);
26356bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
26366bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  node =
26376bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines      (IndexDataStringList *)malloc(sizeof(IndexDataStringList) + strlen(name) +
26386bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                                    digitCount(line) + digitCount(column) + 2);
26396bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  newStr = node->data;
26404e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  sprintf(newStr, "%s:%d:%d", name, line, column);
26416bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
26426bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  /* Remember string so it can be freed later. */
26436bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data = (IndexData *)client_data;
26446bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  node->next = index_data->strings;
26456bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data->strings = node;
26466bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
26476ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  return (CXIdxClientContainer)newStr;
26484e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
26494e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
26502957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidisstatic void printCXIndexContainer(const CXIdxContainerInfo *info) {
26512957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIdxClientContainer container;
26522957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  container = clang_index_getClientContainer(info);
26533e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis  if (!container)
26543e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis    printf("[<<NULL>>]");
26553e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis  else
26563e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis    printf("[%s]", (const char *)container);
26574e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
26584e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2659dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic const char *getEntityKindString(CXIdxEntityKind kind) {
2660dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  switch (kind) {
2661dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Unexposed: return "<<UNEXPOSED>>";
2662dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Typedef: return "typedef";
2663dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Function: return "function";
2664dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Variable: return "variable";
2665dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Field: return "field";
2666dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_EnumConstant: return "enumerator";
2667dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCClass: return "objc-class";
2668dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCProtocol: return "objc-protocol";
2669dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCCategory: return "objc-category";
2670c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  case CXIdxEntity_ObjCInstanceMethod: return "objc-instance-method";
2671c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  case CXIdxEntity_ObjCClassMethod: return "objc-class-method";
2672dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCProperty: return "objc-property";
2673dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCIvar: return "objc-ivar";
2674dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Enum: return "enum";
2675dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Struct: return "struct";
2676dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Union: return "union";
2677dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_CXXClass: return "c++-class";
26782957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXNamespace: return "namespace";
26792957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXNamespaceAlias: return "namespace-alias";
26802957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXStaticVariable: return "c++-static-var";
26812957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXStaticMethod: return "c++-static-method";
26822957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXInstanceMethod: return "c++-instance-method";
26832957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXConstructor: return "constructor";
26842957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXDestructor: return "destructor";
26852957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXConversionFunction: return "conversion-func";
26862957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXTypeAlias: return "type-alias";
268735adca009c65a9578b91ab0bdca13691624e13afDavid Blaikie  case CXIdxEntity_CXXInterface: return "c++-__interface";
26882957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  }
26892957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  assert(0 && "Garbage entity kind");
26902957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  return 0;
26912957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis}
26922957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis
26932957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidisstatic const char *getEntityTemplateKindString(CXIdxEntityCXXTemplateKind kind) {
26942957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  switch (kind) {
26952957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_NonTemplate: return "";
26962957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_Template: return "-template";
26972957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_TemplatePartialSpecialization:
26982957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    return "-template-partial-spec";
26992957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_TemplateSpecialization: return "-template-spec";
2700dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
27016ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  assert(0 && "Garbage entity kind");
27026ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  return 0;
2703dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis}
2704dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2705838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidisstatic const char *getEntityLanguageString(CXIdxEntityLanguage kind) {
2706838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  switch (kind) {
2707838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_None: return "<none>";
2708838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_C: return "C";
2709838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_ObjC: return "ObjC";
2710838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_CXX: return "C++";
2711838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  }
2712838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  assert(0 && "Garbage language kind");
2713838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  return 0;
2714838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis}
2715838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis
2716dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic void printEntityInfo(const char *cb,
2717dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                            CXClientData client_data,
27186ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                            const CXIdxEntityInfo *info) {
27194e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *name;
27204e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
2721643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  unsigned i;
27224e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
27234e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
27244e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2725c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis  if (!info) {
2726c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis    printf("%s: <<NULL>>", cb);
2727c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis    return;
2728c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis  }
2729c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis
2730dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  name = info->name;
27314e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (!name)
27324e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    name = "<anon-tag>";
27334e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
27342957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  printf("%s: kind: %s%s", cb, getEntityKindString(info->kind),
27352957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis         getEntityTemplateKindString(info->templateKind));
2736dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | name: %s", name);
2737dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | USR: %s", info->USR);
2738c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  printf(" | lang: %s", getEntityLanguageString(info->lang));
2739643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis
2740643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  for (i = 0; i != info->numAttributes; ++i) {
2741643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis    const CXIdxAttrInfo *Attr = info->attributes[i];
2742643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis    printf("     <attribute>: ");
2743f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(Attr->cursor, NULL);
2744643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  }
27454e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
27464e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2747b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidisstatic void printBaseClassInfo(CXClientData client_data,
2748b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis                               const CXIdxBaseClassInfo *info) {
2749b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printEntityInfo("     <base>", client_data, info->base);
2750b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printf(" | cursor: ");
2751f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
2752b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printf(" | loc: ");
275313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2754b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis}
2755b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis
2756c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidisstatic void printProtocolList(const CXIdxObjCProtocolRefListInfo *ProtoInfo,
2757c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis                              CXClientData client_data) {
2758c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  unsigned i;
2759c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  for (i = 0; i < ProtoInfo->numProtocols; ++i) {
2760c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printEntityInfo("     <protocol>", client_data,
2761c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis                    ProtoInfo->protocols[i]->protocol);
2762c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf(" | cursor: ");
2763f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(ProtoInfo->protocols[i]->cursor, NULL);
2764c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf(" | loc: ");
276513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    printCXIndexLoc(ProtoInfo->protocols[i]->loc, client_data);
2766c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf("\n");
2767c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  }
2768c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis}
2769c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis
27704e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void index_diagnostic(CXClientData client_data,
2771996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis                             CXDiagnosticSet diagSet, void *reserved) {
27724e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString str;
27734e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *cstr;
2774996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  unsigned numDiags, i;
2775996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  CXDiagnostic diag;
27764e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
27774e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
27784e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
27794e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2780996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  numDiags = clang_getNumDiagnosticsInSet(diagSet);
2781996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  for (i = 0; i != numDiags; ++i) {
2782996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    diag = clang_getDiagnosticInSet(diagSet, i);
2783996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    str = clang_formatDiagnostic(diag, clang_defaultDiagnosticDisplayOptions());
2784996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    cstr = clang_getCString(str);
2785996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    printf("[diagnostic]: %s\n", cstr);
2786996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    clang_disposeString(str);
2787996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis
2788996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    if (getenv("CINDEXTEST_FAILONERROR") &&
2789996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis        clang_getDiagnosticSeverity(diag) >= CXDiagnostic_Error) {
2790996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis      index_data->fail_for_error = 1;
2791996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    }
2792dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
27934e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
27944e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2795dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientFile index_enteredMainFile(CXClientData client_data,
2796dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                                       CXFile file, void *reserved) {
2797dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  IndexData *index_data;
279862d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis  CXString filename;
279962d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis
2800dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
2801dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCheck(index_data);
2802dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
280362d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis  filename = clang_getFileName(file);
280413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  index_data->main_filename = clang_getCString(filename);
280513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  clang_disposeString(filename);
280613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
2807dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf("[enteredMainFile]: ");
2808dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)file);
2809dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf("\n");
2810dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2811dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientFile)file;
28124e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
28134e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2814dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientFile index_ppIncludedFile(CXClientData client_data,
28156ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                            const CXIdxIncludedFileInfo *info) {
28164e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
28176bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  CXModule Mod;
28184e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
28194e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
28204e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
282166042b32b3b37ddcba731ff05c2792e3bb572102Argyrios Kyrtzidis  printf("[ppIncludedFile]: ");
2822dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)info->file);
28234e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | name: \"%s\"", info->filename);
28244e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | hash loc: ");
282513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->hashLoc, client_data);
28266bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  printf(" | isImport: %d | isAngled: %d | isModule: %d",
28278d7a24e94b58676e57fd3f47353cbdbc59917d81Argyrios Kyrtzidis         info->isImport, info->isAngled, info->isModuleImport);
28286bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
28296bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  Mod = clang_getModuleForFile(index_data->TU, (CXFile)info->file);
28306bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  if (Mod) {
28316bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    CXString str = clang_Module_getFullName(Mod);
28326bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    const char *cstr = clang_getCString(str);
28336bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    printf(" | module: %s", cstr);
28346bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    clang_disposeString(str);
28356bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  }
28366bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines
28376bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  printf("\n");
2838dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2839dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientFile)info->file;
28404e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
28414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
28422c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidisstatic CXIdxClientFile index_importedASTFile(CXClientData client_data,
28432c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis                                         const CXIdxImportedASTFileInfo *info) {
28442c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  IndexData *index_data;
28452c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
28462c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  printCheck(index_data);
28472c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis
284811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (index_data->importedASTs) {
284910f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString filename = clang_getFileName(info->file);
285011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    importedASTS_insert(index_data->importedASTs, clang_getCString(filename));
285111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    clang_disposeString(filename);
285211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  }
285311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
28542c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  printf("[importedASTFile]: ");
28552c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)info->file);
2856134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis  if (info->module) {
285710f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString name = clang_Module_getFullName(info->module);
2858134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printf(" | loc: ");
2859134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printCXIndexLoc(info->loc, client_data);
2860134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printf(" | name: \"%s\"", clang_getCString(name));
2861134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printf(" | isImplicit: %d\n", info->isImplicit);
2862134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    clang_disposeString(name);
2863900ab95e12bb7483971640a91075699eec391804Argyrios Kyrtzidis  } else {
28643c5527e939c454bfd3e16354908e0c4315be5d2fNAKAMURA Takumi    /* PCH file, the rest are not relevant. */
2865900ab95e12bb7483971640a91075699eec391804Argyrios Kyrtzidis    printf("\n");
2866134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis  }
28672c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis
28682c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  return (CXIdxClientFile)info->file;
28692c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis}
28702c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis
28716bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesstatic CXIdxClientContainer
28726bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hinesindex_startedTranslationUnit(CXClientData client_data, void *reserved) {
28734e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
28744e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
28754e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
28764e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
287766042b32b3b37ddcba731ff05c2792e3bb572102Argyrios Kyrtzidis  printf("[startedTranslationUnit]\n");
2878dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientContainer)"TU";
28794e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
28804e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
28816ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidisstatic void index_indexDeclaration(CXClientData client_data,
28822957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                                   const CXIdxDeclInfo *info) {
2883dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  IndexData *index_data;
28846ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  const CXIdxObjCCategoryDeclInfo *CatInfo;
28856ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  const CXIdxObjCInterfaceDeclInfo *InterInfo;
2886c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  const CXIdxObjCProtocolRefListInfo *ProtoInfo;
2887792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  const CXIdxObjCPropertyDeclInfo *PropInfo;
2888b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  const CXIdxCXXClassDeclInfo *CXXClassInfo;
2889b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  unsigned i;
2890dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
28914e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2892dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo("[indexDeclaration]", client_data, info->entityInfo);
2893dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | cursor: ");
2894f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
2895dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | loc: ");
289613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2897b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printf(" | semantic-container: ");
2898b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printCXIndexContainer(info->semanticContainer);
2899b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printf(" | lexical-container: ");
2900b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printCXIndexContainer(info->lexicalContainer);
2901dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | isRedecl: %d", info->isRedeclaration);
2902c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isDef: %d", info->isDefinition);
2903838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  if (info->flags & CXIdxDeclFlag_Skipped) {
2904838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    assert(!info->isContainer);
2905838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    printf(" | isContainer: skipped");
2906838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  } else {
2907838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    printf(" | isContainer: %d", info->isContainer);
2908838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  }
2909c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isImplicit: %d\n", info->isImplicit);
29104e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2911b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  for (i = 0; i != info->numAttributes; ++i) {
291287adb0bf9375390de4c66d9e2ad110cc492cd655NAKAMURA Takumi    const CXIdxAttrInfo *Attr = info->attributes[i];
2913b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis    printf("     <attribute>: ");
2914f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(Attr->cursor, NULL);
2915b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis    printf("\n");
2916b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  }
2917b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis
2918dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  if (clang_index_isEntityObjCContainerKind(info->entityInfo->kind)) {
2919dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    const char *kindName = 0;
2920dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    CXIdxObjCContainerKind K = clang_index_getObjCContainerDeclInfo(info)->kind;
2921dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    switch (K) {
2922dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_ForwardRef:
2923dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "forward-ref"; break;
2924dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_Interface:
2925dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "interface"; break;
2926dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_Implementation:
2927dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "implementation"; break;
2928dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    }
2929dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printCheck(index_data);
2930dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printf("     <ObjCContainerInfo>: kind: %s\n", kindName);
2931dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
29324e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
29336ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  if ((CatInfo = clang_index_getObjCCategoryDeclInfo(info))) {
2934dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printEntityInfo("     <ObjCCategoryInfo>: class", client_data,
2935dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                    CatInfo->objcClass);
293621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    printf(" | cursor: ");
2937f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(CatInfo->classCursor, NULL);
293821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    printf(" | loc: ");
293913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    printCXIndexLoc(CatInfo->classLoc, client_data);
2940dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printf("\n");
2941dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
29424e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
29436ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  if ((InterInfo = clang_index_getObjCInterfaceDeclInfo(info))) {
29446ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis    if (InterInfo->superInfo) {
2945b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printBaseClassInfo(client_data, InterInfo->superInfo);
29466ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis      printf("\n");
29476ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis    }
29484e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
29494e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2950c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  if ((ProtoInfo = clang_index_getObjCProtocolRefListInfo(info))) {
2951c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printProtocolList(ProtoInfo, client_data);
29526ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  }
29534e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2954792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  if ((PropInfo = clang_index_getObjCPropertyDeclInfo(info))) {
2955792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    if (PropInfo->getter) {
2956792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printEntityInfo("     <getter>", client_data, PropInfo->getter);
2957792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printf("\n");
2958792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    }
2959792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    if (PropInfo->setter) {
2960792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printEntityInfo("     <setter>", client_data, PropInfo->setter);
2961792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printf("\n");
2962792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    }
2963792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  }
2964792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis
2965b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  if ((CXXClassInfo = clang_index_getCXXClassDeclInfo(info))) {
2966b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis    for (i = 0; i != CXXClassInfo->numBases; ++i) {
2967b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printBaseClassInfo(client_data, CXXClassInfo->bases[i]);
2968b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printf("\n");
2969b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis    }
2970b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  }
2971b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis
29722957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  if (info->declAsContainer)
29736bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    clang_index_setClientContainer(
29746bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        info->declAsContainer,
29756bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines        makeClientContainer(client_data, info->entityInfo, info->loc));
29764e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
29774e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
29784e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void index_indexEntityReference(CXClientData client_data,
29796ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                       const CXIdxEntityRefInfo *info) {
29806bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  printEntityInfo("[indexEntityReference]", client_data,
29816bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines                  info->referencedEntity);
29824e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | cursor: ");
2983f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
29844e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | loc: ");
298513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2986dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo(" | <parent>:", client_data, info->parentEntity);
29874e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | container: ");
29884e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCXIndexContainer(info->container);
2989c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | refkind: ");
2990aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  switch (info->kind) {
2991aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  case CXIdxEntityRef_Direct: printf("direct"); break;
2992b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  case CXIdxEntityRef_Implicit: printf("implicit"); break;
2993aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  }
29944e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("\n");
29954e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
29964e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
29976f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidisstatic int index_abortQuery(CXClientData client_data, void *reserved) {
29986f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  IndexData *index_data;
29996f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
30006f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  return index_data->abort;
30016f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis}
30026f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis
30034e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic IndexerCallbacks IndexCB = {
30046f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_abortQuery,
30054e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_diagnostic,
3006dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_enteredMainFile,
30074e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_ppIncludedFile,
30082c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  index_importedASTFile,
30094e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_startedTranslationUnit,
3010dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_indexDeclaration,
30114e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_indexEntityReference
30124e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis};
30134e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
301422490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidisstatic unsigned getIndexOptions(void) {
301522490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  unsigned index_opts;
301622490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  index_opts = 0;
301722490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  if (getenv("CINDEXTEST_SUPPRESSREFS"))
301822490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis    index_opts |= CXIndexOpt_SuppressRedundantRefs;
301922490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  if (getenv("CINDEXTEST_INDEXLOCALSYMBOLS"))
302022490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis    index_opts |= CXIndexOpt_IndexFunctionLocalSymbols;
3021838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  if (!getenv("CINDEXTEST_DISABLE_SKIPPARSEDBODIES"))
3022838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    index_opts |= CXIndexOpt_SkipParsedBodiesInSession;
302322490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis
302422490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  return index_opts;
302522490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis}
302622490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis
3027d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidisstatic int index_compile_args(int num_args, const char **args,
3028d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              CXIndexAction idxAction,
3029d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              ImportedASTFilesData *importedASTs,
3030d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              const char *check_prefix) {
3031d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  IndexData index_data;
3032d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  unsigned index_opts;
3033d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  int result;
3034d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3035d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (num_args == 0) {
3036d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    fprintf(stderr, "no compiler arguments\n");
3037d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return -1;
3038d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
3039d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3040d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.check_prefix = check_prefix;
3041d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.first_check_printed = 0;
3042d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.fail_for_error = 0;
3043d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.abort = 0;
3044d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.main_filename = "";
3045d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.importedASTs = importedASTs;
30466bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data.strings = NULL;
30476bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data.TU = NULL;
3048d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3049d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_opts = getIndexOptions();
3050d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = clang_indexSourceFile(idxAction, &index_data,
3051d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                 &IndexCB,sizeof(IndexCB), index_opts,
3052d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                 0, args, num_args, 0, 0, 0,
3053d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                 getDefaultParsingOptions());
3054651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (result != CXError_Success)
3055651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(result);
3056651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
3057d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (index_data.fail_for_error)
3058d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    result = -1;
3059d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
30606bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  free_client_data(&index_data);
3061d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return result;
3062d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis}
3063d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3064d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidisstatic int index_ast_file(const char *ast_file,
3065d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          CXIndex Idx,
3066d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          CXIndexAction idxAction,
3067d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          ImportedASTFilesData *importedASTs,
3068d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          const char *check_prefix) {
3069d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  CXTranslationUnit TU;
3070d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  IndexData index_data;
3071d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  unsigned index_opts;
3072d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  int result;
3073d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3074d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (!CreateTranslationUnit(Idx, ast_file, &TU))
3075d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return -1;
3076d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3077d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.check_prefix = check_prefix;
3078d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.first_check_printed = 0;
3079d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.fail_for_error = 0;
3080d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.abort = 0;
3081d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.main_filename = "";
3082d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.importedASTs = importedASTs;
30836bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data.strings = NULL;
30846bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  index_data.TU = TU;
3085d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3086d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_opts = getIndexOptions();
3087d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = clang_indexTranslationUnit(idxAction, &index_data,
3088d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                      &IndexCB,sizeof(IndexCB),
3089d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                      index_opts, TU);
3090d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (index_data.fail_for_error)
3091d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    result = -1;
3092d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3093d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  clang_disposeTranslationUnit(TU);
30946bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines  free_client_data(&index_data);
3095d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return result;
3096d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis}
3097d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
309811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic int index_file(int argc, const char **argv, int full) {
30994e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *check_prefix;
31002957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndex Idx;
31012957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndexAction idxAction;
3102d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  ImportedASTFilesData *importedASTs;
3103dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  int result;
31044e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
31054e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  check_prefix = 0;
31064e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc > 0) {
31074e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
31084e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
31094e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      ++argv;
31104e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      --argc;
31114e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    }
31124e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
31134e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
31142957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
3115fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                                /* displayDiagnostics=*/1))) {
31162957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
31172957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    return 1;
31182957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  }
3119d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
3120d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  importedASTs = 0;
312111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (full)
3122d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    importedASTs = importedASTs_create();
3123d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3124d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = index_compile_args(argc, argv, idxAction, importedASTs, check_prefix);
3125d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (result != 0)
3126d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    goto finished;
3127dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
312811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (full) {
312911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    unsigned i;
3130d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    for (i = 0; i < importedASTs->num_files && result == 0; ++i) {
3131d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      result = index_ast_file(importedASTs->filenames[i], Idx, idxAction,
3132d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              importedASTs, check_prefix);
313311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    }
313411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  }
313521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
313611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisfinished:
3137d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  importedASTs_dispose(importedASTs);
31382957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
31392957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_disposeIndex(Idx);
314021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  return result;
314121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis}
314221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
314321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidisstatic int index_tu(int argc, const char **argv) {
3144d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  const char *check_prefix;
314521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  CXIndex Idx;
31462957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndexAction idxAction;
314721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  int result;
314821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
314921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  check_prefix = 0;
315021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc > 0) {
315121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
315221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
315321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      ++argv;
315421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      --argc;
315521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    }
315621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
315721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
3158d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
3159fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                                /* displayDiagnostics=*/1))) {
3160d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
3161d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return 1;
3162d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
3163d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
3164d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3165d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = index_ast_file(argv[0], Idx, idxAction,
3166d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          /*importedASTs=*/0, check_prefix);
3167d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3168d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
3169d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  clang_disposeIndex(Idx);
3170d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return result;
3171d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis}
3172d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3173d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidisstatic int index_compile_db(int argc, const char **argv) {
3174d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  const char *check_prefix;
3175d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  CXIndex Idx;
3176d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  CXIndexAction idxAction;
3177d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  int errorCode = 0;
3178d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3179d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  check_prefix = 0;
3180d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (argc > 0) {
3181d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
3182d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
3183d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      ++argv;
3184d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      --argc;
3185d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    }
3186d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
3187d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
318821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc == 0) {
3189d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    fprintf(stderr, "no compilation database\n");
319021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return -1;
319121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
319221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
319321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
3194fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit                                /* displayDiagnostics=*/1))) {
319521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
319621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return 1;
319721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
3198d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
319921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
3200d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  {
3201d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    const char *database = argv[0];
3202d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompilationDatabase db = 0;
3203d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompileCommands CCmds = 0;
3204d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompileCommand CCmd;
3205d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompilationDatabase_Error ec;
3206d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXString wd;
3207d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#define MAX_COMPILE_ARGS 512
3208d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXString cxargs[MAX_COMPILE_ARGS];
3209d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    const char *args[MAX_COMPILE_ARGS];
3210d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    char *tmp;
3211d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    unsigned len;
3212d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    char *buildDir;
3213d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    int i, a, numCmds, numArgs;
3214d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3215d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    len = strlen(database);
3216d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    tmp = (char *) malloc(len+1);
3217d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    memcpy(tmp, database, len+1);
3218d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    buildDir = dirname(tmp);
3219d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3220d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    db = clang_CompilationDatabase_fromDirectory(buildDir, &ec);
3221d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3222d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    if (db) {
3223d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3224d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      if (ec!=CXCompilationDatabase_NoError) {
3225d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        printf("unexpected error %d code while loading compilation database\n", ec);
3226d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = -1;
3227d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        goto cdb_end;
3228d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
322921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
32302bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis      if (chdir(buildDir) != 0) {
32312bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        printf("Could not chdir to %s\n", buildDir);
32322bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        errorCode = -1;
32332bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        goto cdb_end;
32342bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis      }
323521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
32362bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis      CCmds = clang_CompilationDatabase_getAllCompileCommands(db);
3237d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      if (!CCmds) {
3238d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        printf("compilation db is empty\n");
3239d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = -1;
3240d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        goto cdb_end;
3241d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
3242d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3243d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      numCmds = clang_CompileCommands_getSize(CCmds);
3244d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3245d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      if (numCmds==0) {
3246d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        fprintf(stderr, "should not get an empty compileCommand set\n");
3247d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = -1;
3248d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        goto cdb_end;
3249d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
3250d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3251d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      for (i=0; i<numCmds && errorCode == 0; ++i) {
3252d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        CCmd = clang_CompileCommands_getCommand(CCmds, i);
3253d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3254d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        wd = clang_CompileCommand_getDirectory(CCmd);
32552bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        if (chdir(clang_getCString(wd)) != 0) {
32562bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis          printf("Could not chdir to %s\n", clang_getCString(wd));
32572bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis          errorCode = -1;
32582bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis          goto cdb_end;
32592bff7e5a02495dabd26013c15613f082fce67633Argyrios Kyrtzidis        }
3260d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        clang_disposeString(wd);
3261d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3262d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        numArgs = clang_CompileCommand_getNumArgs(CCmd);
3263d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        if (numArgs > MAX_COMPILE_ARGS){
3264d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          fprintf(stderr, "got more compile arguments than maximum\n");
3265d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          errorCode = -1;
3266d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          goto cdb_end;
3267d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        }
3268d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        for (a=0; a<numArgs; ++a) {
3269d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          cxargs[a] = clang_CompileCommand_getArg(CCmd, a);
3270d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          args[a] = clang_getCString(cxargs[a]);
3271d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        }
3272d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3273d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = index_compile_args(numArgs, args, idxAction,
3274d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                       /*importedASTs=*/0, check_prefix);
3275d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3276d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        for (a=0; a<numArgs; ++a)
3277d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          clang_disposeString(cxargs[a]);
3278d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
3279d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    } else {
3280d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      printf("database loading failed with error code %d.\n", ec);
3281d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      errorCode = -1;
3282d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    }
3283d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3284d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  cdb_end:
3285d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    clang_CompileCommands_dispose(CCmds);
3286d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    clang_CompilationDatabase_dispose(db);
3287d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    free(tmp);
3288d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
3289d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
32904e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
32912957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
32922957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_disposeIndex(Idx);
3293d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return errorCode;
32944e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
32954e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
3296fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregorint perform_token_annotation(int argc, const char **argv) {
3297fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  const char *input = argv[1];
3298fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  char *filename = 0;
3299fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned line, second_line;
3300fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned column, second_column;
3301fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXIndex CIdx;
3302fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXTranslationUnit TU = 0;
3303fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  int errorCode;
3304fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
3305fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  int num_unsaved_files = 0;
3306fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXToken *tokens;
3307fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned num_tokens;
3308fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXSourceRange range;
3309fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXSourceLocation startLoc, endLoc;
3310fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXFile file = 0;
3311fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXCursor *cursors = 0;
3312651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  CXSourceRangeList *skipped_ranges = 0;
3313651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
3314fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned i;
3315fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3316fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  input += strlen("-test-annotate-tokens=");
3317fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if ((errorCode = parse_file_line_column(input, &filename, &line, &column,
3318fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                          &second_line, &second_column)))
3319fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return errorCode;
3320fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3321e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  if (parse_remapped_files(argc, argv, 2, &unsaved_files, &num_unsaved_files)) {
3322e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    free(filename);
3323fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return -1;
3324e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  }
3325fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
33260a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  CIdx = clang_createIndex(0, 1);
3327651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(CIdx, argv[argc - 1],
3328651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argv + num_unsaved_files + 2,
3329651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    argc - num_unsaved_files - 3,
3330651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    unsaved_files,
3331651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    num_unsaved_files,
3332651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                    getDefaultParsingOptions(), &TU);
3333651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
3334fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    fprintf(stderr, "unable to parse input\n");
3335651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
3336fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    clang_disposeIndex(CIdx);
3337fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    free(filename);
3338fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
3339fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return -1;
3340e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
3341fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  errorCode = 0;
3342fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3343e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  if (checkForErrors(TU) != 0) {
3344e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    errorCode = -1;
3345e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    goto teardown;
3346e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  }
3347dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3348ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING")) {
3349ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    for (i = 0; i < 5; ++i) {
3350651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Err = clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
3351651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                                         clang_defaultReparseOptions(TU));
3352651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      if (Err != CXError_Success) {
3353ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        fprintf(stderr, "Unable to reparse translation unit!\n");
3354651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines        describeLibclangFailure(Err);
3355ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        errorCode = -1;
3356ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        goto teardown;
3357ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      }
3358ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    }
3359ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
3360ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
3361dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
3362dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
3363dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
3364dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
3365dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3366fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  file = clang_getFile(TU, filename);
3367fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!file) {
3368fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    fprintf(stderr, "file %s is not in this translation unit\n", filename);
3369fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
3370fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    goto teardown;
3371fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
3372fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3373fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  startLoc = clang_getLocation(TU, file, line, column);
3374fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (clang_equalLocations(clang_getNullLocation(), startLoc)) {
3375e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(stderr, "invalid source location %s:%d:%d\n", filename, line,
3376fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            column);
3377fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
3378e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    goto teardown;
3379fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
3380fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3381fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  endLoc = clang_getLocation(TU, file, second_line, second_column);
3382fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (clang_equalLocations(clang_getNullLocation(), endLoc)) {
3383e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(stderr, "invalid source location %s:%d:%d\n", filename,
3384fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            second_line, second_column);
3385fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
3386e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    goto teardown;
3387fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
3388fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3389fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  range = clang_getRange(startLoc, endLoc);
3390fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_tokenize(TU, range, &tokens, &num_tokens);
3391dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3392dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
3393dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
3394dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
3395dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
3396dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3397fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  cursors = (CXCursor *)malloc(num_tokens * sizeof(CXCursor));
3398fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_annotateTokens(TU, tokens, num_tokens, cursors);
3399dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3400dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
3401dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
3402dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
3403dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
3404dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
3405651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  skipped_ranges = clang_getSkippedRanges(TU, file);
3406651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  for (i = 0; i != skipped_ranges->count; ++i) {
3407651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    unsigned start_line, start_column, end_line, end_column;
3408651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_getSpellingLocation(clang_getRangeStart(skipped_ranges->ranges[i]),
3409651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                              0, &start_line, &start_column, 0);
3410651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_getSpellingLocation(clang_getRangeEnd(skipped_ranges->ranges[i]),
3411651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines                              0, &end_line, &end_column, 0);
3412651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    printf("Skipping: ");
3413651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    PrintExtent(stdout, start_line, start_column, end_line, end_column);
3414651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    printf("\n");
3415651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  }
3416651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  clang_disposeSourceRangeList(skipped_ranges);
3417651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
3418fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  for (i = 0; i != num_tokens; ++i) {
3419fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *kind = "<unknown>";
342010f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXString spelling = clang_getTokenSpelling(TU, tokens[i]);
342110f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceRange extent = clang_getTokenExtent(TU, tokens[i]);
3422fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    unsigned start_line, start_column, end_line, end_column;
3423fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3424fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    switch (clang_getTokenKind(tokens[i])) {
3425fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Punctuation: kind = "Punctuation"; break;
3426fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Keyword: kind = "Keyword"; break;
3427fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Identifier: kind = "Identifier"; break;
3428fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Literal: kind = "Literal"; break;
3429fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Comment: kind = "Comment"; break;
3430fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
3431a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(clang_getRangeStart(extent),
3432a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              0, &start_line, &start_column, 0);
3433a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(clang_getRangeEnd(extent),
3434a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              0, &end_line, &end_column, 0);
343551b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar    printf("%s: \"%s\" ", kind, clang_getCString(spelling));
3436342742adb69af8ef8f2c652f162496b98c45bc20Benjamin Kramer    clang_disposeString(spelling);
343751b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar    PrintExtent(stdout, start_line, start_column, end_line, end_column);
34380045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    if (!clang_isInvalid(cursors[i].kind)) {
34390045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor      printf(" ");
3440f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko      PrintCursor(cursors[i], NULL);
34410045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    }
34420045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    printf("\n");
3443fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
3444fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free(cursors);
344593f5e6a5d7690f90bc8a94e6b40d6c7d19719e0cTed Kremenek  clang_disposeTokens(TU, tokens, num_tokens);
3446fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3447fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor teardown:
3448a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
3449fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_disposeTranslationUnit(TU);
3450fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_disposeIndex(CIdx);
3451fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free(filename);
3452fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
3453fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  return errorCode;
3454fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
3455fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3456db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonstatic int
3457db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonperform_test_compilation_db(const char *database, int argc, const char **argv) {
3458db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompilationDatabase db;
3459db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompileCommands CCmds;
3460db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompileCommand CCmd;
3461db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompilationDatabase_Error ec;
3462db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXString wd;
3463db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXString arg;
3464db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  int errorCode = 0;
3465db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  char *tmp;
3466db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  unsigned len;
3467db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  char *buildDir;
3468db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  int i, j, a, numCmds, numArgs;
3469db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3470db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  len = strlen(database);
3471db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  tmp = (char *) malloc(len+1);
3472db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  memcpy(tmp, database, len+1);
3473db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  buildDir = dirname(tmp);
3474db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3475c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison  db = clang_CompilationDatabase_fromDirectory(buildDir, &ec);
3476db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3477db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  if (db) {
3478db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3479db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    if (ec!=CXCompilationDatabase_NoError) {
3480db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      printf("unexpected error %d code while loading compilation database\n", ec);
3481db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      errorCode = -1;
3482db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      goto cdb_end;
3483db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    }
3484db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3485db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    for (i=0; i<argc && errorCode==0; ) {
3486db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      if (strcmp(argv[i],"lookup")==0){
3487c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        CCmds = clang_CompilationDatabase_getCompileCommands(db, argv[i+1]);
3488db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3489db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (!CCmds) {
3490db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("file %s not found in compilation db\n", argv[i+1]);
3491db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          errorCode = -1;
3492db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          break;
3493db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
3494db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3495c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        numCmds = clang_CompileCommands_getSize(CCmds);
3496db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3497db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (numCmds==0) {
3498db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          fprintf(stderr, "should not get an empty compileCommand set for file"
3499db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison                          " '%s'\n", argv[i+1]);
3500db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          errorCode = -1;
3501db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          break;
3502db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
3503db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3504db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        for (j=0; j<numCmds; ++j) {
3505c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          CCmd = clang_CompileCommands_getCommand(CCmds, j);
3506db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3507c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          wd = clang_CompileCommand_getDirectory(CCmd);
3508db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("workdir:'%s'", clang_getCString(wd));
3509db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          clang_disposeString(wd);
3510db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3511db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf(" cmdline:'");
3512c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          numArgs = clang_CompileCommand_getNumArgs(CCmd);
3513db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          for (a=0; a<numArgs; ++a) {
3514db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            if (a) printf(" ");
3515c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison            arg = clang_CompileCommand_getArg(CCmd, a);
3516db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            printf("%s", clang_getCString(arg));
3517db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            clang_disposeString(arg);
3518db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          }
3519db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("'\n");
3520db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
3521db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3522c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        clang_CompileCommands_dispose(CCmds);
3523db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3524db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        i += 2;
3525db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      }
3526db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    }
3527c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison    clang_CompilationDatabase_dispose(db);
3528db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  } else {
3529db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    printf("database loading failed with error code %d.\n", ec);
3530db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    errorCode = -1;
3531db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  }
3532db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3533db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisoncdb_end:
3534db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  free(tmp);
3535db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3536db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  return errorCode;
3537db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison}
3538db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
35390d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
3540f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/* USR printing.                                                              */
3541f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/******************************************************************************/
3542f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3543f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic int insufficient_usr(const char *kind, const char *usage) {
3544f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "USR for '%s' requires: %s\n", kind, usage);
3545f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 1;
3546f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3547f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3548f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic unsigned isUSR(const char *s) {
3549f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return s[0] == 'c' && s[1] == ':';
3550f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3551f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3552f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic int not_usr(const char *s, const char *arg) {
3553f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "'%s' argument ('%s') is not a USR\n", s, arg);
3554f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 1;
3555f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3556f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3557f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic void print_usr(CXString usr) {
3558f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  const char *s = clang_getCString(usr);
3559f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  printf("%s\n", s);
3560f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  clang_disposeString(usr);
3561f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3562f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3563f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic void display_usrs() {
3564f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "-print-usrs options:\n"
3565f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCCategory <class name> <category name>\n"
3566f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCClass <class name>\n"
3567f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCIvar <ivar name> <class USR>\n"
3568f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCMethod <selector> [0=class method|1=instance method] "
3569f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            "<class USR>\n"
3570f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          " ObjCProperty <property name> <class USR>\n"
3571f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          " ObjCProtocol <protocol name>\n");
3572f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3573f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3574f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekint print_usrs(const char **I, const char **E) {
3575f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  while (I != E) {
3576f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    const char *kind = *I;
3577f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    unsigned len = strlen(kind);
3578f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    switch (len) {
3579f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 8:
3580f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCIvar", 8) == 0) {
3581f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
3582f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<ivar name> <class USR>");
3583f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[2]))
3584f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[2]);
3585f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
3586f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
3587a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[2];
3588ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
3589f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCIvar(I[1], x));
3590f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
3591f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3592f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
3593f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3594f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3595f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3596f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 9:
3597f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCClass", 9) == 0) {
3598f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 1 >= E)
3599f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<class name>");
3600f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCClass(I[1]));
3601f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 2;
3602f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3603f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3604f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3605f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 10:
3606f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCMethod", 10) == 0) {
3607f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 3 >= E)
3608f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<method selector> "
3609f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek                "[0=class method|1=instance method] <class USR>");
3610f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[3]))
3611f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[3]);
3612f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
3613f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
3614a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[3];
3615ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
3616f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCMethod(I[1], atoi(I[2]), x));
3617f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
3618f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 4;
3619f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3620f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3621f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3622f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 12:
3623f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCCategory", 12) == 0) {
3624f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
3625f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<class name> <category name>");
3626f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCCategory(I[1], I[2]));
3627f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
3628f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3629f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3630f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCProtocol", 12) == 0) {
3631f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 1 >= E)
3632f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<protocol name>");
3633f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCProtocol(I[1]));
3634f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 2;
3635f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3636f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3637f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCProperty", 12) == 0) {
3638f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
3639f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<property name> <class USR>");
3640f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[2]))
3641f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[2]);
3642f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
3643f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
3644a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[2];
3645ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
3646f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCProperty(I[1], x));
3647f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
3648f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
3649f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3650f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3651f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3652f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      default:
3653f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3654f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
3655f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    break;
3656f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3657f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3658f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  if (I != E) {
3659f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    fprintf(stderr, "Invalid USR kind: %s\n", *I);
3660f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    display_usrs();
3661f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return 1;
3662f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3663f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 0;
3664f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3665f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3666f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekint print_usrs_file(const char *file_name) {
3667f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  char line[2048];
3668f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  const char *args[128];
3669f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  unsigned numChars = 0;
3670f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3671f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  FILE *fp = fopen(file_name, "r");
3672f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  if (!fp) {
3673f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    fprintf(stderr, "error: cannot open '%s'\n", file_name);
3674f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return 1;
3675f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3676f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3677f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  /* This code is not really all that safe, but it works fine for testing. */
3678f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  while (!feof(fp)) {
3679f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    char c = fgetc(fp);
3680f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    if (c == '\n') {
3681f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      unsigned i = 0;
3682f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      const char *s = 0;
3683f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3684f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (numChars == 0)
3685f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        continue;
3686f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3687f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      line[numChars] = '\0';
3688f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      numChars = 0;
3689f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3690f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (line[0] == '/' && line[1] == '/')
3691f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        continue;
3692f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3693f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      s = strtok(line, " ");
3694f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      while (s) {
3695f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        args[i] = s;
3696f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        ++i;
3697f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        s = strtok(0, " ");
3698f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      }
3699f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (print_usrs(&args[0], &args[i]))
3700f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        return 1;
3701f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
3702f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    else
3703f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      line[numChars++] = c;
3704f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3705f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3706f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fclose(fp);
3707f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 0;
3708f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3709f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3710f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/******************************************************************************/
37110d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Command line processing.                                                   */
37120d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
37137ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregorint write_pch_file(const char *filename, int argc, const char *argv[]) {
37147ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  CXIndex Idx;
37157ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  CXTranslationUnit TU;
37167ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
37177ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  int num_unsaved_files = 0;
3718651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  enum CXErrorCode Err;
371908aa622ab923fb8a2751f213feee0b9124ee86fbFrancois Pichet  int result = 0;
37207ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
3721fc09336a5965040736f9bc63a70416003972364eStefanus Du Toit  Idx = clang_createIndex(/* excludeDeclsFromPCH */1, /* displayDiagnostics=*/1);
37227ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
37237ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
37247ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    clang_disposeIndex(Idx);
37257ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return -1;
37267ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  }
3727651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
3728651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  Err = clang_parseTranslationUnit2(
3729651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      Idx, 0, argv + num_unsaved_files, argc - num_unsaved_files,
3730651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      unsaved_files, num_unsaved_files,
3731651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      CXTranslationUnit_Incomplete |
3732651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          CXTranslationUnit_DetailedPreprocessingRecord |
3733651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines          CXTranslationUnit_ForSerialization,
3734651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines      &TU);
3735651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  if (Err != CXError_Success) {
37367ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
3737651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    describeLibclangFailure(Err);
37387ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
3739651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    clang_disposeTranslationUnit(TU);
37407ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    clang_disposeIndex(Idx);
37417ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return 1;
37427ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  }
37437ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
374439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  switch (clang_saveTranslationUnit(TU, filename,
374539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor                                    clang_defaultSaveOptions(TU))) {
374639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_None:
374739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
374839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
374939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_TranslationErrors:
375039c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: translation errors\n",
375139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor            filename);
375239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 2;
375339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
375439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
375539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_InvalidTU:
375639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: invalid translation unit\n",
375739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor            filename);
375839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 3;
375939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
376039c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
376139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_Unknown:
376239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  default:
376339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: unknown error \n", filename);
376439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 1;
376539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
376639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  }
376739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
37687ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  clang_disposeTranslationUnit(TU);
37697ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
37707ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  clang_disposeIndex(Idx);
377139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  return result;
37727ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor}
37737ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
37747ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/******************************************************************************/
3775153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/* Serialized diagnostics.                                                    */
3776153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/******************************************************************************/
3777153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3778153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic const char *getDiagnosticCodeStr(enum CXLoadDiag_Error error) {
3779153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  switch (error) {
3780153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_CannotLoad: return "Cannot Load File";
3781153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_None: break;
3782153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_Unknown: return "Unknown";
3783153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_InvalidFile: return "Invalid File";
3784153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3785153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return "None";
3786153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3787153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3788153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic const char *getSeverityString(enum CXDiagnosticSeverity severity) {
3789153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  switch (severity) {
3790153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Note: return "note";
3791153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Error: return "error";
3792153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Fatal: return "fatal";
3793153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Ignored: return "ignored";
3794153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Warning: return "warning";
3795153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3796153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return "unknown";
3797153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3798153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3799153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printIndent(unsigned indent) {
3800a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  if (indent == 0)
3801a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek    return;
3802a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  fprintf(stderr, "+");
3803a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  --indent;
3804153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  while (indent > 0) {
3805a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek    fprintf(stderr, "-");
3806153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    --indent;
3807153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3808153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3809153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3810153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printLocation(CXSourceLocation L) {
3811153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXFile File;
3812153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXString FileName;
3813153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned line, column, offset;
3814153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3815153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_getExpansionLocation(L, &File, &line, &column, &offset);
3816153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  FileName = clang_getFileName(File);
3817153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3818153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  fprintf(stderr, "%s:%d:%d", clang_getCString(FileName), line, column);
3819153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_disposeString(FileName);
3820153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3821153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3822153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printRanges(CXDiagnostic D, unsigned indent) {
3823153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned i, n = clang_getDiagnosticNumRanges(D);
3824153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3825153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0; i < n; ++i) {
3826153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceLocation Start, End;
382710f900447e41a1a1a1877b886553da4a0dff3ad6Enea Zaffanella    CXSourceRange SR = clang_getDiagnosticRange(D, i);
3828153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    Start = clang_getRangeStart(SR);
3829153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    End = clang_getRangeEnd(SR);
3830153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3831153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3832153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "Range: ");
3833153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(Start);
3834153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, " ");
3835153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(End);
3836153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "\n");
3837153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3838153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3839153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3840153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printFixIts(CXDiagnostic D, unsigned indent) {
3841153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned i, n = clang_getDiagnosticNumFixIts(D);
38423739b32a01b9778176757425e62252c288b85383Ted Kremenek  fprintf(stderr, "Number FIXITs = %d\n", n);
3843153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0 ; i < n; ++i) {
3844153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceRange ReplacementRange;
3845153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXString text;
3846153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    text = clang_getDiagnosticFixIt(D, i, &ReplacementRange);
3847153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3848153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3849153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "FIXIT: (");
3850153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(clang_getRangeStart(ReplacementRange));
3851153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, " - ");
3852153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(clang_getRangeEnd(ReplacementRange));
3853153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "): \"%s\"\n", clang_getCString(text));
3854153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(text);
3855153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3856153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3857153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3858153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printDiagnosticSet(CXDiagnosticSet Diags, unsigned indent) {
38599190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi  unsigned i, n;
38609190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi
3861153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (!Diags)
3862153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    return;
3863153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
38649190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi  n = clang_getNumDiagnosticsInSet(Diags);
3865153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0; i < n; ++i) {
3866153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceLocation DiagLoc;
3867153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXDiagnostic D;
3868153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXFile File;
386978d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    CXString FileName, DiagSpelling, DiagOption, DiagCat;
3870153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    unsigned line, column, offset;
387178d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    const char *DiagOptionStr = 0, *DiagCatStr = 0;
3872153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3873153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    D = clang_getDiagnosticInSet(Diags, i);
3874153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagLoc = clang_getDiagnosticLocation(D);
3875153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_getExpansionLocation(DiagLoc, &File, &line, &column, &offset);
3876153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    FileName = clang_getFileName(File);
3877153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagSpelling = clang_getDiagnosticSpelling(D);
3878153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3879153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3880153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3881153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "%s:%d:%d: %s: %s",
3882153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(FileName),
3883153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            line,
3884153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            column,
3885153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            getSeverityString(clang_getDiagnosticSeverity(D)),
3886153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(DiagSpelling));
3887153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3888153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagOption = clang_getDiagnosticOption(D, 0);
3889153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagOptionStr = clang_getCString(DiagOption);
3890153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    if (DiagOptionStr) {
3891153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek      fprintf(stderr, " [%s]", DiagOptionStr);
3892153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    }
3893153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
389478d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    DiagCat = clang_getDiagnosticCategoryText(D);
389578d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    DiagCatStr = clang_getCString(DiagCat);
389678d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    if (DiagCatStr) {
389778d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek      fprintf(stderr, " [%s]", DiagCatStr);
389878d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    }
389978d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek
3900153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "\n");
3901153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3902153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printRanges(D, indent);
3903153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printFixIts(D, indent);
3904153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3905a4ca95a6ebe85d63df08f3290372e97ddaf0b02aNAKAMURA Takumi    /* Print subdiagnostics. */
3906153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printDiagnosticSet(clang_getChildDiagnostics(D), indent+2);
3907153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3908153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(FileName);
3909153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(DiagSpelling);
3910153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(DiagOption);
39116bcf27bb9a4b5c3f79cb44c0e4654a6d7619ad89Stephen Hines    clang_disposeString(DiagCat);
3912153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3913153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3914153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3915153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic int read_diagnostics(const char *filename) {
3916153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  enum CXLoadDiag_Error error;
3917153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXString errorString;
3918153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXDiagnosticSet Diags = 0;
3919153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3920153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  Diags = clang_loadDiagnostics(filename, &error, &errorString);
3921153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (!Diags) {
3922153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "Trouble deserializing file (%s): %s\n",
3923153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            getDiagnosticCodeStr(error),
3924153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(errorString));
3925153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(errorString);
3926153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    return 1;
3927153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3928153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3929153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  printDiagnosticSet(Diags, 0);
3930a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  fprintf(stderr, "Number of diagnostics: %d\n",
3931a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek          clang_getNumDiagnosticsInSet(Diags));
3932153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_disposeDiagnosticSet(Diags);
3933153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return 0;
3934153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3935153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3936651f13cea278ec967336033dd032faef0e9fc2ecStephen Hinesstatic int perform_print_build_session_timestamp(void) {
3937651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  printf("%lld\n", clang_getBuildSessionTimestamp());
3938651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  return 0;
3939651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines}
3940651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines
3941153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/******************************************************************************/
39427ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/* Command line processing.                                                   */
39437ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/******************************************************************************/
3944f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek
3945e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregorstatic CXCursorVisitor GetVisitor(const char *s) {
39467d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  if (s[0] == '\0')
3947e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return FilteredPrintingVisitor;
39487d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  if (strcmp(s, "-usrs") == 0)
39497d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    return USRVisitor;
395059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  if (strncmp(s, "-memory-usage", 13) == 0)
395159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    return GetVisitor(s + 13);
39527d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  return NULL;
39537d40562f83552b7295411e10ee887d8d55470679Ted Kremenek}
39547d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
3955f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenekstatic void print_usage(void) {
3956f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  fprintf(stderr,
39570d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "usage: c-index-test -code-completion-at=<site> <compiler arguments>\n"
39581982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -code-completion-timing=<site> <compiler arguments>\n"
3959f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    "       c-index-test -cursor-at=<site> <compiler arguments>\n"
3960ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    "       c-index-test -file-refs-at=<site> <compiler arguments>\n"
3961ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    "       c-index-test -file-includes-in=<filename> <compiler arguments>\n");
39623584972866f39d42aa4029586278022d89451bd9NAKAMURA Takumi  fprintf(stderr,
39634e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    "       c-index-test -index-file [-check-prefix=<FileCheck prefix>] <compiler arguments>\n"
396411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    "       c-index-test -index-file-full [-check-prefix=<FileCheck prefix>] <compiler arguments>\n"
396521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    "       c-index-test -index-tu [-check-prefix=<FileCheck prefix>] <AST file>\n"
3966d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    "       c-index-test -index-compile-db [-check-prefix=<FileCheck prefix>] <compilation database>\n"
39671d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    "       c-index-test -test-file-scan <AST file> <source file> "
396826fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen          "[FileCheck prefix]\n");
396926fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen  fprintf(stderr,
3970fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek    "       c-index-test -test-load-tu <AST file> <symbol filter> "
3971fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek          "[FileCheck prefix]\n"
39727d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    "       c-index-test -test-load-tu-usrs <AST file> <symbol filter> "
39737d40562f83552b7295411e10ee887d8d55470679Ted Kremenek           "[FileCheck prefix]\n"
39741982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -test-load-source <symbol filter> {<args>}*\n");
3975f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  fprintf(stderr,
397659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-load-source-memory-usage "
397759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "<symbol filter> {<args>}*\n"
3978abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    "       c-index-test -test-load-source-reparse <trials> <symbol filter> "
3979abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    "          {<args>}*\n"
39801982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -test-load-source-usrs <symbol filter> {<args>}*\n"
398159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-load-source-usrs-memory-usage "
398259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek          "<symbol filter> {<args>}*\n"
398316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    "       c-index-test -test-annotate-tokens=<range> {<args>}*\n"
398416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    "       c-index-test -test-inclusion-stack-source {<args>}*\n"
39854e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    "       c-index-test -test-inclusion-stack-tu <AST file>\n");
398653513d2e24205965ffd5d4780a10f41cc335f17bChandler Carruth  fprintf(stderr,
39874e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    "       c-index-test -test-print-linkage-source {<args>}*\n"
3988ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko    "       c-index-test -test-print-type {<args>}*\n"
3989411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    "       c-index-test -test-print-type-size {<args>}*\n"
39901eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    "       c-index-test -test-print-bitwidth {<args>}*\n"
399159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -print-usr [<CursorKind> {<args>}]*\n"
39927ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    "       c-index-test -print-usr-file <file>\n"
3993153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    "       c-index-test -write-pch <file> <compiler arguments>\n");
3994153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  fprintf(stderr,
3995db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    "       c-index-test -compilation-db [lookup <filename>] database\n");
3996db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  fprintf(stderr,
3997651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    "       c-index-test -print-build-session-timestamp\n");
3998651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  fprintf(stderr,
3999153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    "       c-index-test -read-diagnostics <file>\n\n");
4000caf4bd3d3c66fd95b0159356e2c4da7fbf2897f3Douglas Gregor  fprintf(stderr,
40017d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    " <symbol filter> values:\n%s",
40020d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   all - load all symbols, including those from PCH\n"
40030d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   local - load all symbols except those in PCH\n"
40040d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   category - only load ObjC categories (non-PCH)\n"
40050d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   interface - only load ObjC interfaces (non-PCH)\n"
40060d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   protocol - only load ObjC protocols (non-PCH)\n"
40070d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   function - only load functions (non-PCH)\n"
4008625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    "   typedef - only load typdefs (non-PCH)\n"
4009625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    "   scan-function - scan function bodies (non-PCH)\n\n");
4010f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek}
4011f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek
40126edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/***/
40136edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
40146edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbarint cindextest_main(int argc, const char **argv) {
40150a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  clang_enableStackTraces();
4016153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (argc > 2 && strcmp(argv[1], "-read-diagnostics") == 0)
4017153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek      return read_diagnostics(argv[2]);
4018f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  if (argc > 2 && strstr(argv[1], "-code-completion-at=") == argv[1])
40191982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    return perform_code_completion(argc, argv, 0);
40201982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  if (argc > 2 && strstr(argv[1], "-code-completion-timing=") == argv[1])
40211982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    return perform_code_completion(argc, argv, 1);
4022f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  if (argc > 2 && strstr(argv[1], "-cursor-at=") == argv[1])
4023f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return inspect_cursor_at(argc, argv);
4024aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (argc > 2 && strstr(argv[1], "-file-refs-at=") == argv[1])
4025aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return find_file_refs_at(argc, argv);
4026ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis  if (argc > 2 && strstr(argv[1], "-file-includes-in=") == argv[1])
4027ee2d5fd7a3bd42bce387db094d27a479617c3e67Argyrios Kyrtzidis    return find_file_includes_in(argc, argv);
40284e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-file") == 0)
402911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    return index_file(argc - 2, argv + 2, /*full=*/0);
403011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-file-full") == 0)
403111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    return index_file(argc - 2, argv + 2, /*full=*/1);
403221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-tu") == 0)
403321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return index_tu(argc - 2, argv + 2);
4034d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-compile-db") == 0)
4035d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return index_compile_db(argc - 2, argv + 2);
40367d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strncmp(argv[1], "-test-load-tu", 13) == 0) {
4037e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 13);
40387d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    if (I)
4039ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek      return perform_test_load_tu(argv[2], argv[3], argc >= 5 ? argv[4] : 0, I,
4040ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                                  NULL);
40417d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  }
4042abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  else if (argc >= 5 && strncmp(argv[1], "-test-load-source-reparse", 25) == 0){
4043abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 25);
4044abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    if (I) {
4045abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      int trials = atoi(argv[2]);
4046abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      return perform_test_reparse_source(argc - 4, argv + 4, trials, argv[3], I,
4047abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                         NULL);
4048abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    }
4049abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
40507d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strncmp(argv[1], "-test-load-source", 17) == 0) {
4051e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 17);
405259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
405359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    PostVisitTU postVisit = 0;
405459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    if (strstr(argv[1], "-memory-usage"))
405559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      postVisit = PrintMemoryUsage;
405659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
40577d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    if (I)
405859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      return perform_test_load_source(argc - 3, argv + 3, argv[2], I,
405959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek                                      postVisit);
40607d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  }
40617d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strcmp(argv[1], "-test-file-scan") == 0)
40621d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    return perform_file_scan(argv[2], argv[3],
40631d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                             argc >= 5 ? argv[4] : 0);
4064fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  else if (argc > 2 && strstr(argv[1], "-test-annotate-tokens=") == argv[1])
4065fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return perform_token_annotation(argc, argv);
406616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-inclusion-stack-source") == 0)
406716b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all", NULL,
406816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                                    PrintInclusionStack);
406916b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-inclusion-stack-tu") == 0)
407016b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    return perform_test_load_tu(argv[2], "all", NULL, NULL,
407116b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                                PrintInclusionStack);
40723bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-print-linkage-source") == 0)
40733bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all", PrintLinkage,
40743bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek                                    NULL);
4075ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko  else if (argc > 2 && strcmp(argv[1], "-test-print-type") == 0)
40768e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all",
4077ae03d8e52d81adbb98b31dd5c179abe45c91fc25Dmitri Gribenko                                    PrintType, 0);
4078411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis  else if (argc > 2 && strcmp(argv[1], "-test-print-type-size") == 0)
4079411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis    return perform_test_load_source(argc - 2, argv + 2, "all",
4080411d33aa0b0d3bc9b2faec40cd821bdd836094abArgyrios Kyrtzidis                                    PrintTypeSize, 0);
40811eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  else if (argc > 2 && strcmp(argv[1], "-test-print-bitwidth") == 0)
40821eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    return perform_test_load_source(argc - 2, argv + 2, "all",
40831eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko                                    PrintBitWidth, 0);
4084f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  else if (argc > 1 && strcmp(argv[1], "-print-usr") == 0) {
4085f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    if (argc > 2)
4086f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      return print_usrs(argv + 2, argv + argc);
4087f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    else {
4088f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      display_usrs();
4089f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      return 1;
4090f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
4091f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
4092f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-print-usr-file") == 0)
4093f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return print_usrs_file(argv[2]);
40947ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  else if (argc > 2 && strcmp(argv[1], "-write-pch") == 0)
40957ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return write_pch_file(argv[2], argc - 3, argv + 3);
4096db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  else if (argc > 2 && strcmp(argv[1], "-compilation-db") == 0)
4097db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    return perform_test_compilation_db(argv[argc-1], argc - 3, argv + 2);
4098651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines  else if (argc == 2 && strcmp(argv[1], "-print-build-session-timestamp") == 0)
4099651f13cea278ec967336033dd032faef0e9fc2ecStephen Hines    return perform_print_build_session_timestamp();
4100db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
4101f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  print_usage();
410250398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff  return 1;
410350398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff}
41046edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
41056edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/***/
41066edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
41076edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/* We intentionally run in a separate thread to ensure we at least minimal
41086edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar * testing of a multithreaded environment (for example, having a reduced stack
41096edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar * size). */
41106edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
41116edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbartypedef struct thread_info {
41126edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  int argc;
41136edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  const char **argv;
41146edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  int result;
41156edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar} thread_info;
411684294911b83003bcac726d00ffa91143752feb57Benjamin Kramervoid thread_runner(void *client_data_v) {
41176edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  thread_info *client_data = client_data_v;
41186edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data->result = cindextest_main(client_data->argc, client_data->argv);
4119ef8225444452a1486bd721f3285301fe84643b00Stephen Hines}
4120ef8225444452a1486bd721f3285301fe84643b00Stephen Hines
4121ef8225444452a1486bd721f3285301fe84643b00Stephen Hinesstatic void flush_atexit(void) {
4122ef8225444452a1486bd721f3285301fe84643b00Stephen Hines  /* stdout, and surprisingly even stderr, are not always flushed on process
4123ef8225444452a1486bd721f3285301fe84643b00Stephen Hines   * and thread exit, particularly when the system is under heavy load. */
4124ef8225444452a1486bd721f3285301fe84643b00Stephen Hines  fflush(stdout);
4125ef8225444452a1486bd721f3285301fe84643b00Stephen Hines  fflush(stderr);
41266edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar}
41276edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
41286edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbarint main(int argc, const char **argv) {
4129d1a4f68a4301d1ee3098cc9db0cd507b96dd1beeBenjamin Kramer  thread_info client_data;
4130d1a4f68a4301d1ee3098cc9db0cd507b96dd1beeBenjamin Kramer
4131ef8225444452a1486bd721f3285301fe84643b00Stephen Hines  atexit(flush_atexit);
4132ef8225444452a1486bd721f3285301fe84643b00Stephen Hines
4133f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
4134f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  LIBXML_TEST_VERSION
4135f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
4136f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
41376160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor  if (getenv("CINDEXTEST_NOTHREADS"))
41386160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor    return cindextest_main(argc, argv);
41396160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor
41406edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data.argc = argc;
41416edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data.argv = argv;
4142a32a6e1abe6972b48b4d5ec79d81b6f7241723bfDaniel Dunbar  clang_executeOnThread(thread_runner, &client_data, 0);
41436edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  return client_data.result;
41446edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar}
4145