c-index-test.c revision 838eb7e8652e451d93494a4e583e4d11809bcb4a
12b8ee6c2994f738e5162ff46b638974870f51662Steve Naroff/* c-index-test.c */
250398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
350398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff#include "clang-c/Index.h"
4db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison#include "clang-c/CXCompilationDatabase.h"
5f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include "llvm/Config/config.h"
61e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor#include <ctype.h>
70c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor#include <stdlib.h>
889922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff#include <stdio.h>
9af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff#include <string.h>
10f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor#include <assert.h>
11af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff
12f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
13f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include <libxml/parser.h>
14f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include <libxml/relaxng.h>
15f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#include <libxml/xmlerror.h>
16f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
17f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
18d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#ifdef _WIN32
19d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#  include <direct.h>
20d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#else
21d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#  include <unistd.h>
22d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#endif
23d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
240d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
250d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Utility functions.                                                         */
260d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
270d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
282e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#ifdef _MSC_VER
292e06fc877a633abea3b40a64950c7316dac29ca8John Thompsonchar *basename(const char* path)
302e06fc877a633abea3b40a64950c7316dac29ca8John Thompson{
312e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    char* base1 = (char*)strrchr(path, '/');
322e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    char* base2 = (char*)strrchr(path, '\\');
332e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    if (base1 && base2)
342e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return((base1 > base2) ? base1 + 1 : base2 + 1);
352e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    else if (base1)
362e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return(base1 + 1);
372e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    else if (base2)
382e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return(base2 + 1);
392e06fc877a633abea3b40a64950c7316dac29ca8John Thompson
402e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    return((char*)path);
412e06fc877a633abea3b40a64950c7316dac29ca8John Thompson}
42db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonchar *dirname(char* path)
43db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison{
44db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    char* base1 = (char*)strrchr(path, '/');
45db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    char* base2 = (char*)strrchr(path, '\\');
46db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    if (base1 && base2)
47db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (base1 > base2)
48db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          *base1 = 0;
49db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        else
50db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          *base2 = 0;
51db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    else if (base1)
520fb474a9323525964d437fefa22c06172492d270NAKAMURA Takumi        *base1 = 0;
53db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    else if (base2)
540fb474a9323525964d437fefa22c06172492d270NAKAMURA Takumi        *base2 = 0;
55db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
56db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    return path;
57db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison}
582e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#else
59ff9e18cb3863defa98257500ef49d93a9d2bc216Steve Naroffextern char *basename(const char *);
60db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonextern char *dirname(char *);
612e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#endif
62ff9e18cb3863defa98257500ef49d93a9d2bc216Steve Naroff
6345ba9a1b31110568d0d362c8d31c6133cf9011b7Douglas Gregor/** \brief Return the default parsing options. */
6444c181aec37789f25f6c15543c164416f72e562aDouglas Gregorstatic unsigned getDefaultParsingOptions() {
6544c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  unsigned options = CXTranslationUnit_DetailedPreprocessingRecord;
6644c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
6744c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  if (getenv("CINDEXTEST_EDITING"))
68b1c031be513705d924038f497279b9b599868ba1Douglas Gregor    options |= clang_defaultEditingTranslationUnitOptions();
6987c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor  if (getenv("CINDEXTEST_COMPLETION_CACHING"))
7087c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor    options |= CXTranslationUnit_CacheCompletionResults;
71dcaca015a2f08e03a6d86222fec2e83e04f6b848Argyrios Kyrtzidis  if (getenv("CINDEXTEST_COMPLETION_NO_CACHING"))
72dcaca015a2f08e03a6d86222fec2e83e04f6b848Argyrios Kyrtzidis    options &= ~CXTranslationUnit_CacheCompletionResults;
736a91d385618ea4d28236c496f540a26877c95525Erik Verbruggen  if (getenv("CINDEXTEST_SKIP_FUNCTION_BODIES"))
746a91d385618ea4d28236c496f540a26877c95525Erik Verbruggen    options |= CXTranslationUnit_SkipFunctionBodies;
75d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (getenv("CINDEXTEST_COMPLETION_BRIEF_COMMENTS"))
76d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    options |= CXTranslationUnit_IncludeBriefCommentsInCodeCompletion;
7744c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
7844c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  return options;
7944c181aec37789f25f6c15543c164416f72e562aDouglas Gregor}
8044c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
81bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidisstatic int checkForErrors(CXTranslationUnit TU);
82bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
8351b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbarstatic void PrintExtent(FILE *out, unsigned begin_line, unsigned begin_column,
8451b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar                        unsigned end_line, unsigned end_column) {
8551b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  fprintf(out, "[%d:%d - %d:%d]", begin_line, begin_column,
86d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar          end_line, end_column);
8751b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar}
8851b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar
891c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenekstatic unsigned CreateTranslationUnit(CXIndex Idx, const char *file,
901c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek                                      CXTranslationUnit *TU) {
91e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
92a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  *TU = clang_createTranslationUnit(Idx, file);
936be2a22979e6a4af575af32cccdf6a6190a8edcaDan Gohman  if (!*TU) {
941c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Unable to load translation unit from '%s'!\n", file);
951c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 0;
96e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
971c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  return 1;
981c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
991c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
1004db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregorvoid free_remapped_files(struct CXUnsavedFile *unsaved_files,
1014db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                         int num_unsaved_files) {
1024db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int i;
1034db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (i = 0; i != num_unsaved_files; ++i) {
1044db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    free((char *)unsaved_files[i].Filename);
1054db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    free((char *)unsaved_files[i].Contents);
1064db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
107653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor  free(unsaved_files);
1084db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor}
1094db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
1104db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregorint parse_remapped_files(int argc, const char **argv, int start_arg,
1114db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                         struct CXUnsavedFile **unsaved_files,
1124db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                         int *num_unsaved_files) {
1134db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int i;
1144db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int arg;
1154db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int prefix_len = strlen("-remap-file=");
1164db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *unsaved_files = 0;
1174db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *num_unsaved_files = 0;
118e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1194db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  /* Count the number of remapped files. */
1204db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (arg = start_arg; arg < argc; ++arg) {
1214db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (strncmp(argv[arg], "-remap-file=", prefix_len))
1224db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      break;
123e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1244db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    ++*num_unsaved_files;
1254db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
126e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1274db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  if (*num_unsaved_files == 0)
1284db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    return 0;
129e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1304db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *unsaved_files
131653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor    = (struct CXUnsavedFile *)malloc(sizeof(struct CXUnsavedFile) *
132653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor                                     *num_unsaved_files);
1334db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (arg = start_arg, i = 0; i != *num_unsaved_files; ++i, ++arg) {
1344db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    struct CXUnsavedFile *unsaved = *unsaved_files + i;
1354db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    const char *arg_string = argv[arg] + prefix_len;
1364db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    int filename_len;
1374db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    char *filename;
1384db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    char *contents;
1394db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    FILE *to_file;
1404db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    const char *semi = strchr(arg_string, ';');
1414db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (!semi) {
142e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr,
1434db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor              "error: -remap-file=from;to argument is missing semicolon\n");
1444db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
1454db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
1464db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
1474db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
1484db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
149e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1504db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Open the file that we're remapping to. */
151c44fe4bf2962bfcb323e93a63ba4a72d67839cdeFrancois Pichet    to_file = fopen(semi + 1, "rb");
1524db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (!to_file) {
1534db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fprintf(stderr, "error: cannot open file %s that we are remapping to\n",
1544db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor              semi + 1);
1554db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
1564db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
1574db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
1584db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
1594db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
160e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1614db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Determine the length of the file we're remapping to. */
1624db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fseek(to_file, 0, SEEK_END);
1634db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Length = ftell(to_file);
1644db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fseek(to_file, 0, SEEK_SET);
165e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1664db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Read the contents of the file we're remapping to. */
1674db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    contents = (char *)malloc(unsaved->Length + 1);
1684db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (fread(contents, 1, unsaved->Length, to_file) != unsaved->Length) {
1694db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fprintf(stderr, "error: unexpected %s reading 'to' file %s\n",
1704db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor              (feof(to_file) ? "EOF" : "error"), semi + 1);
1714db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fclose(to_file);
1724db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
173e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith      free(contents);
1744db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
1754db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
1764db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
1774db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
1784db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    contents[unsaved->Length] = 0;
1794db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Contents = contents;
180e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1814db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Close the file. */
1824db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fclose(to_file);
183e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1844db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Copy the file name that we're remapping from. */
1854db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    filename_len = semi - arg_string;
1864db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    filename = (char *)malloc(filename_len + 1);
1874db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    memcpy(filename, arg_string, filename_len);
1884db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    filename[filename_len] = 0;
1894db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Filename = filename;
1904db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
191e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1924db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  return 0;
1934db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor}
1944db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
195f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkostatic const char *parse_comments_schema(int argc, const char **argv) {
196f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentsSchemaArg = "-comments-xml-schema=";
197f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentSchemaFile = NULL;
198f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
199f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (argc == 0)
200f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return CommentSchemaFile;
201f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
202f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!strncmp(argv[0], CommentsSchemaArg, strlen(CommentsSchemaArg)))
203f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    CommentSchemaFile = argv[0] + strlen(CommentsSchemaArg);
204f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
205f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  return CommentSchemaFile;
206f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko}
207f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
2080d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
2090d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Pretty-printing.                                                           */
2100d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
2110d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
212ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic const char *FileCheckPrefix = "CHECK";
213ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
214ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCString(const char *CStr) {
2152d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  if (CStr != NULL && CStr[0] != '\0') {
2162d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko    for ( ; *CStr; ++CStr) {
2172d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      const char C = *CStr;
2182d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      switch (C) {
2192d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\n': printf("\\n"); break;
2202d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\r': printf("\\r"); break;
2212d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\t': printf("\\t"); break;
2222d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\v': printf("\\v"); break;
2232d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\f': printf("\\f"); break;
2242d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        default:   putchar(C);    break;
2252d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      }
2262d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko    }
2272d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  }
228ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
229ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
230ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCStringWithPrefix(const char *Prefix, const char *CStr) {
231ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf(" %s=[", Prefix);
232ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCString(CStr);
2332d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  printf("]");
2342d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko}
2352d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko
236ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCXStringAndDispose(CXString Str) {
237ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCString(clang_getCString(Str));
238ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  clang_disposeString(Str);
239ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
240ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
241f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkostatic void PrintCXStringWithPrefix(const char *Prefix, CXString Str) {
242f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCStringWithPrefix(Prefix, clang_getCString(Str));
243f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko}
244f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
245ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCXStringWithPrefixAndDispose(const char *Prefix,
246ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                              CXString Str) {
247ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCStringWithPrefix(Prefix, clang_getCString(Str));
248ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  clang_disposeString(Str);
249ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
250ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
251430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregorstatic void PrintRange(CXSourceRange R, const char *str) {
252430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  CXFile begin_file, end_file;
253430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  unsigned begin_line, begin_column, end_line, end_column;
254430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
255430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  clang_getSpellingLocation(clang_getRangeStart(R),
256430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                            &begin_file, &begin_line, &begin_column, 0);
257430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  clang_getSpellingLocation(clang_getRangeEnd(R),
258430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                            &end_file, &end_line, &end_column, 0);
259430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  if (!begin_file || !end_file)
260430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    return;
261430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
262ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis  if (str)
263ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis    printf(" %s=", str);
264430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  PrintExtent(stdout, begin_line, begin_column, end_line, end_column);
265430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor}
266430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
267358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregorint want_display_name = 0;
268358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
269cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregorstatic void printVersion(const char *Prefix, CXVersion Version) {
270cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Major < 0)
271cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
272cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf("%s%d", Prefix, Version.Major);
273cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
274cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Minor < 0)
275cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
276cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf(".%d", Version.Minor);
277cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
278cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Subminor < 0)
279cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
280cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf(".%d", Version.Subminor);
281cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor}
282cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
283ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostruct CommentASTDumpingContext {
284ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  int IndentLevel;
285ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko};
286ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
287ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void DumpCXCommentInternal(struct CommentASTDumpingContext *Ctx,
288ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                  CXComment Comment) {
2895ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  unsigned i;
2905ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  unsigned e;
2915ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  enum CXCommentKind Kind = clang_Comment_getKind(Comment);
2925ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko
293ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx->IndentLevel++;
2945ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  for (i = 0, e = Ctx->IndentLevel; i != e; ++i)
295ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("  ");
296ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
297ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("(");
298ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  switch (Kind) {
299ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Null:
300ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Null");
301ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
302ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Text:
303ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Text");
304ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose("Text",
305ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                      clang_TextComment_getText(Comment));
306ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_isWhitespace(Comment))
307ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" IsWhitespace");
308ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
309ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
310ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
311ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_InlineCommand:
312ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_InlineCommand");
313ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
314ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
315ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_InlineCommandComment_getCommandName(Comment));
3162d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    switch (clang_InlineCommandComment_getRenderKind(Comment)) {
3172d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Normal:
3182d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderNormal");
3192d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
3202d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Bold:
3212d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderBold");
3222d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
3232d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Monospaced:
3242d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderMonospaced");
3252d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
3262d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Emphasized:
3272d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderEmphasized");
3282d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
3292d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    }
3305ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0, e = clang_InlineCommandComment_getNumArgs(Comment);
331ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko         i != e; ++i) {
332ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Arg[%u]=", i);
333ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringAndDispose(
334ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko          clang_InlineCommandComment_getArgText(Comment, i));
335ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
336ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
337ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
338ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
3395ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  case CXComment_HTMLStartTag: {
3405ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    unsigned NumAttrs;
341ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_HTMLStartTag");
342ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
343ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Name",
344ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_HTMLTagComment_getTagName(Comment));
3455ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    NumAttrs = clang_HTMLStartTag_getNumAttrs(Comment);
346ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (NumAttrs != 0) {
347ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Attrs:");
3485ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko      for (i = 0; i != NumAttrs; ++i) {
349ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        printf(" ");
350ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCXStringAndDispose(clang_HTMLStartTag_getAttrName(Comment, i));
351ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        printf("=");
352ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCXStringAndDispose(clang_HTMLStartTag_getAttrValue(Comment, i));
353ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      }
354ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
355ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_HTMLStartTagComment_isSelfClosing(Comment))
356ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" SelfClosing");
357ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
358ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
359ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
3605ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  }
361ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_HTMLEndTag:
362ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_HTMLEndTag");
363ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
364ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Name",
365ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_HTMLTagComment_getTagName(Comment));
366ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
367ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
368ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
369ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Paragraph:
370ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Paragraph");
371ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_isWhitespace(Comment))
372ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" IsWhitespace");
373ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
374ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_BlockCommand:
375ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_BlockCommand");
376ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
377ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
378ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_BlockCommandComment_getCommandName(Comment));
3795ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0, e = clang_BlockCommandComment_getNumArgs(Comment);
380ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko         i != e; ++i) {
381ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Arg[%u]=", i);
382ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringAndDispose(
383ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko          clang_BlockCommandComment_getArgText(Comment, i));
384ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
385ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
386ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_ParamCommand:
387ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_ParamCommand");
388ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    switch (clang_ParamCommandComment_getDirection(Comment)) {
389ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_In:
390ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" in");
391ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
392ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_Out:
393ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" out");
394ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
395ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_InOut:
396ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" in,out");
397ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
398ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
399ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_ParamCommandComment_isDirectionExplicit(Comment))
400ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" explicitly");
401ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    else
402ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" implicitly");
403ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
404ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "ParamName",
405ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_ParamCommandComment_getParamName(Comment));
406ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_ParamCommandComment_isParamIndexValid(Comment))
407ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" ParamIndex=%u", clang_ParamCommandComment_getParamIndex(Comment));
408ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    else
409ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" ParamIndex=Invalid");
410ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
41196b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko  case CXComment_TParamCommand:
41296b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    printf("CXComment_TParamCommand");
41396b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    PrintCXStringWithPrefixAndDispose(
41496b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        "ParamName",
41596b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        clang_TParamCommandComment_getParamName(Comment));
41696b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    if (clang_TParamCommandComment_isParamPositionValid(Comment)) {
41796b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf(" ParamPosition={");
41896b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      for (i = 0, e = clang_TParamCommandComment_getDepth(Comment);
41996b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko           i != e; ++i) {
42096b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        printf("%u", clang_TParamCommandComment_getIndex(Comment, i));
42196b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        if (i != e - 1)
42296b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko          printf(", ");
42396b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      }
42496b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf("}");
42596b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    } else
42696b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf(" ParamPosition=Invalid");
42796b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    break;
428ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimBlockCommand:
429ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimBlockCommand");
430ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
431ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
432ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_BlockCommandComment_getCommandName(Comment));
433ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
434ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimBlockLine:
435ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimBlockLine");
436ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
437ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Text",
438ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_VerbatimBlockLineComment_getText(Comment));
439ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
440ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimLine:
441ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimLine");
442ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
443ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Text",
444ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_VerbatimLineComment_getText(Comment));
445ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
446ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_FullComment:
447ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_FullComment");
448ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
449ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
450ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  if (Kind != CXComment_Null) {
451ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const unsigned NumChildren = clang_Comment_getNumChildren(Comment);
4525ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    unsigned i;
4535ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0; i != NumChildren; ++i) {
454ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf("\n// %s: ", FileCheckPrefix);
455ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      DumpCXCommentInternal(Ctx, clang_Comment_getChild(Comment, i));
456ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
457ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
458ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf(")");
459ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx->IndentLevel--;
460ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
461ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
462ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void DumpCXComment(CXComment Comment) {
463ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  struct CommentASTDumpingContext Ctx;
464ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx.IndentLevel = 1;
465ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("\n// %s:  CommentAST=[\n// %s:", FileCheckPrefix, FileCheckPrefix);
466ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  DumpCXCommentInternal(&Ctx, Comment);
467ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("]");
468ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
469ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
470f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkotypedef struct {
471f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentSchemaFile;
472f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
473f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGParserCtxtPtr RNGParser;
474f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGPtr Schema;
475f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
476f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko} CommentXMLValidationData;
477f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
478f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkostatic void ValidateCommentXML(const char *Str,
479f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                               CommentXMLValidationData *ValidationData) {
480f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
481f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlDocPtr Doc;
482f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGValidCtxtPtr ValidationCtxt;
483f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  int status;
484f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
485f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!ValidationData || !ValidationData->CommentSchemaFile)
486f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return;
487f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
488f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!ValidationData->RNGParser) {
489f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    ValidationData->RNGParser =
490f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        xmlRelaxNGNewParserCtxt(ValidationData->CommentSchemaFile);
491f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    ValidationData->Schema = xmlRelaxNGParse(ValidationData->RNGParser);
492f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
493f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!ValidationData->RNGParser) {
494f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" libXMLError");
495f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return;
496f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
497f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
498f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  Doc = xmlParseDoc((const xmlChar *) Str);
499f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
500f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!Doc) {
501f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    xmlErrorPtr Error = xmlGetLastError();
502f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" CommentXMLInvalid [not well-formed XML: %s]", Error->message);
503f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    return;
504f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
505f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
506f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  ValidationCtxt = xmlRelaxNGNewValidCtxt(ValidationData->Schema);
507f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  status = xmlRelaxNGValidateDoc(ValidationCtxt, Doc);
508f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if (!status)
509f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" CommentXMLValid");
510f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  else if (status > 0) {
511f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    xmlErrorPtr Error = xmlGetLastError();
512f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" CommentXMLInvalid [not vaild XML: %s]", Error->message);
513f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  } else
514f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    printf(" libXMLError");
515f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
516f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlRelaxNGFreeValidCtxt(ValidationCtxt);
517f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  xmlFreeDoc(Doc);
518f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
519f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko}
520f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
521e4330a302ac20b41b9800267ebd4b5b01f8553f8Dmitri Gribenkostatic void PrintCursorComments(CXCursor Cursor,
522f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                                CommentXMLValidationData *ValidationData) {
523ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  {
524ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    CXString RawComment;
525ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const char *RawCommentCString;
526ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    CXString BriefComment;
527ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const char *BriefCommentCString;
528ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
529ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    RawComment = clang_Cursor_getRawCommentText(Cursor);
530ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    RawCommentCString = clang_getCString(RawComment);
531ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (RawCommentCString != NULL && RawCommentCString[0] != '\0') {
532ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCStringWithPrefix("RawComment", RawCommentCString);
533ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintRange(clang_Cursor_getCommentRange(Cursor), "RawCommentRange");
534ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
535ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      BriefComment = clang_Cursor_getBriefCommentText(Cursor);
536ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      BriefCommentCString = clang_getCString(BriefComment);
537ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      if (BriefCommentCString != NULL && BriefCommentCString[0] != '\0')
538ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCStringWithPrefix("BriefComment", BriefCommentCString);
539ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      clang_disposeString(BriefComment);
540ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
541ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    clang_disposeString(RawComment);
542ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
543ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
544ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  {
545ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    CXComment Comment = clang_Cursor_getParsedComment(Cursor);
546ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_getKind(Comment) != CXComment_Null) {
547ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringWithPrefixAndDispose("FullCommentAsHTML",
548ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                        clang_FullComment_getAsHTML(Comment));
549f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko      {
550f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        CXString XML;
551e4330a302ac20b41b9800267ebd4b5b01f8553f8Dmitri Gribenko        XML = clang_FullComment_getAsXML(Comment);
552f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCXStringWithPrefix("FullCommentAsXML", XML);
553f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        ValidateCommentXML(clang_getCString(XML), ValidationData);
554f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        clang_disposeString(XML);
555f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko      }
556f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
557ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      DumpCXComment(Comment);
558ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
559ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
560ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
561ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
562b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidistypedef struct {
563b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  unsigned line;
564b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  unsigned col;
565b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis} LineCol;
566b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis
567b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidisstatic int lineCol_cmp(const void *p1, const void *p2) {
568b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  const LineCol *lhs = p1;
569b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  const LineCol *rhs = p2;
570b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  if (lhs->line != rhs->line)
571b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis    return (int)lhs->line - (int)rhs->line;
572b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis  return (int)lhs->col - (int)rhs->col;
573b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis}
574b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis
575f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkostatic void PrintCursor(CXCursor Cursor,
576f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                        CommentXMLValidationData *ValidationData) {
577aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXTranslationUnit TU = clang_Cursor_getTranslationUnit(Cursor);
578e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (clang_isInvalid(Cursor.kind)) {
579e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    CXString ks = clang_getCursorKindSpelling(Cursor.kind);
580e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("Invalid Cursor => %s", clang_getCString(ks));
581e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    clang_disposeString(ks);
582e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
583699a07d8a0b1579c5178b3baf4bcf9edb6b38108Steve Naroff  else {
584e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    CXString string, ks;
585c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    CXCursor Referenced;
5861db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    unsigned line, column;
587e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    CXCursor SpecializationOf;
5889f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    CXCursor *overridden;
5899f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    unsigned num_overridden;
590430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    unsigned RefNameRangeNr;
591430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CXSourceRange CursorExtent;
592430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CXSourceRange RefNameRange;
593cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int AlwaysUnavailable;
594cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int AlwaysDeprecated;
595cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXString UnavailableMessage;
596cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXString DeprecatedMessage;
597cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXPlatformAvailability PlatformAvailability[2];
598cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int NumPlatformAvailability;
599cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int I;
600aa0cd85838f2a024e589ea4e8c2094130065af21Dmitri Gribenko
601e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    ks = clang_getCursorKindSpelling(Cursor.kind);
602358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    string = want_display_name? clang_getCursorDisplayName(Cursor)
603358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                              : clang_getCursorSpelling(Cursor);
604e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("%s=%s", clang_getCString(ks),
605e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                    clang_getCString(string));
606e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    clang_disposeString(ks);
607ef0cef6cec8de5fc60e469a93436eed7212e0dc2Steve Naroff    clang_disposeString(string);
608e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
609c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    Referenced = clang_getCursorReferenced(Cursor);
610c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    if (!clang_equalCursors(Referenced, clang_getNullCursor())) {
6111f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (clang_getCursorKind(Referenced) == CXCursor_OverloadedDeclRef) {
6121f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        unsigned I, N = clang_getNumOverloadedDecls(Referenced);
6131f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf("[");
6141f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        for (I = 0; I != N; ++I) {
6151f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          CXCursor Ovl = clang_getOverloadedDecl(Referenced, I);
6161f6206ed21e026ca7ddaf0bff9599c476301a695Douglas Gregor          CXSourceLocation Loc;
6171f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          if (I)
6181f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor            printf(", ");
6191f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
6201f6206ed21e026ca7ddaf0bff9599c476301a695Douglas Gregor          Loc = clang_getCursorLocation(Ovl);
621a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor          clang_getSpellingLocation(Loc, 0, &line, &column, 0);
6221f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          printf("%d:%d", line, column);
6231f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        }
6241f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf("]");
6251f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      } else {
6261f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        CXSourceLocation Loc = clang_getCursorLocation(Referenced);
627a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor        clang_getSpellingLocation(Loc, 0, &line, &column, 0);
6281f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf(":%d:%d", line, column);
6291f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      }
630c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    }
631b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
632b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (clang_isCursorDefinition(Cursor))
633b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      printf(" (Definition)");
63458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
63558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    switch (clang_getCursorAvailability(Cursor)) {
63658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_Available:
63758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
63858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
63958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_Deprecated:
64058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        printf(" (deprecated)");
64158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
64258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
64358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_NotAvailable:
64458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        printf(" (unavailable)");
64558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
646d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen
647d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen      case CXAvailability_NotAccessible:
648d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen        printf(" (inaccessible)");
649d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen        break;
65058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    }
65195f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
652cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    NumPlatformAvailability
653cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      = clang_getCursorPlatformAvailability(Cursor,
654cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &AlwaysDeprecated,
655cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &DeprecatedMessage,
656cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &AlwaysUnavailable,
657cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &UnavailableMessage,
658cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            PlatformAvailability, 2);
659cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    if (AlwaysUnavailable) {
660cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      printf("  (always unavailable: \"%s\")",
661cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor             clang_getCString(UnavailableMessage));
662cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    } else if (AlwaysDeprecated) {
663cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      printf("  (always deprecated: \"%s\")",
664cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor             clang_getCString(DeprecatedMessage));
665cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    } else {
666cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      for (I = 0; I != NumPlatformAvailability; ++I) {
667cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (I >= 2)
668cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          break;
669cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
670cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        printf("  (%s", clang_getCString(PlatformAvailability[I].Platform));
671cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (PlatformAvailability[I].Unavailable)
672cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printf(", unavailable");
673cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        else {
674cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", introduced=", PlatformAvailability[I].Introduced);
675cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", deprecated=", PlatformAvailability[I].Deprecated);
676cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", obsoleted=", PlatformAvailability[I].Obsoleted);
677cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        }
678cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (clang_getCString(PlatformAvailability[I].Message)[0])
679cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printf(", message=\"%s\"",
680cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                 clang_getCString(PlatformAvailability[I].Message));
681cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        printf(")");
682cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      }
683cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    }
684cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    for (I = 0; I != NumPlatformAvailability; ++I) {
685cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      if (I >= 2)
686cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        break;
687cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      clang_disposeCXPlatformAvailability(PlatformAvailability + I);
688cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    }
689cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
690cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    clang_disposeString(DeprecatedMessage);
691cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    clang_disposeString(UnavailableMessage);
692cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
693b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor    if (clang_CXXMethod_isStatic(Cursor))
694b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor      printf(" (static)");
695b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor    if (clang_CXXMethod_isVirtual(Cursor))
696b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor      printf(" (virtual)");
697b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor
69895f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek    if (Cursor.kind == CXCursor_IBOutletCollectionAttr) {
69995f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      CXType T =
70095f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek        clang_getCanonicalType(clang_getIBOutletCollectionType(Cursor));
70195f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      CXString S = clang_getTypeKindSpelling(T.kind);
70295f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      printf(" [IBOutletCollection=%s]", clang_getCString(S));
70395f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      clang_disposeString(S);
70495f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek    }
7053064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
7063064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    if (Cursor.kind == CXCursor_CXXBaseSpecifier) {
7073064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      enum CX_CXXAccessSpecifier access = clang_getCXXAccessSpecifier(Cursor);
7083064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      unsigned isVirtual = clang_isVirtualBase(Cursor);
7093064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      const char *accessStr = 0;
7103064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
7113064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      switch (access) {
7123064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXInvalidAccessSpecifier:
7133064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "invalid"; break;
7143064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXPublic:
7153064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "public"; break;
7163064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXProtected:
7173064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "protected"; break;
7183064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXPrivate:
7193064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "private"; break;
7203064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      }
7213064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
7223064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      printf(" [access=%s isVirtual=%s]", accessStr,
7233064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek             isVirtual ? "true" : "false");
7243064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
725e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor
726e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    SpecializationOf = clang_getSpecializedCursorTemplate(Cursor);
727e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    if (!clang_equalCursors(SpecializationOf, clang_getNullCursor())) {
728e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      CXSourceLocation Loc = clang_getCursorLocation(SpecializationOf);
729e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      CXString Name = clang_getCursorSpelling(SpecializationOf);
730a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor      clang_getSpellingLocation(Loc, 0, &line, &column, 0);
731e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      printf(" [Specialization of %s:%d:%d]",
732e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor             clang_getCString(Name), line, column);
733e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      clang_disposeString(Name);
734e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    }
7359f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
7369f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    clang_getOverriddenCursors(Cursor, &overridden, &num_overridden);
7379f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    if (num_overridden) {
7389f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      unsigned I;
739b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      LineCol lineCols[50];
740b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      assert(num_overridden <= 50);
7419f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      printf(" [Overrides ");
7429f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      for (I = 0; I != num_overridden; ++I) {
7439f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor        CXSourceLocation Loc = clang_getCursorLocation(overridden[I]);
744a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor        clang_getSpellingLocation(Loc, 0, &line, &column, 0);
745b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        lineCols[I].line = line;
746b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        lineCols[I].col = column;
747b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      }
748642214936430d2258318f2022184c1709dfa16d3Michael Liao      /* Make the order of the override list deterministic. */
749b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      qsort(lineCols, num_overridden, sizeof(LineCol), lineCol_cmp);
750b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis      for (I = 0; I != num_overridden; ++I) {
7519f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor        if (I)
7529f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor          printf(", ");
753b3dd988abdd8d048ac999a5713239c95692e528cArgyrios Kyrtzidis        printf("@%d:%d", lineCols[I].line, lineCols[I].col);
7549f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      }
7559f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      printf("]");
7569f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      clang_disposeOverriddenCursors(overridden);
7579f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    }
758ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
759ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    if (Cursor.kind == CXCursor_InclusionDirective) {
760ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      CXFile File = clang_getIncludedFile(Cursor);
761ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      CXString Included = clang_getFileName(File);
762ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      printf(" (%s)", clang_getCString(Included));
763ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      clang_disposeString(Included);
764dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor
765dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor      if (clang_isFileMultipleIncludeGuarded(TU, File))
766dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor        printf("  [multi-include guarded]");
767ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    }
768430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
769430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CursorExtent = clang_getCursorExtent(Cursor);
770430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    RefNameRange = clang_getCursorReferenceNameRange(Cursor,
771430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                   CXNameRange_WantQualifier
772430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantSinglePiece
773430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantTemplateArgs,
774430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                     0);
775430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (!clang_equalRanges(CursorExtent, RefNameRange))
776430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      PrintRange(RefNameRange, "SingleRefName");
777430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
778430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    for (RefNameRangeNr = 0; 1; RefNameRangeNr++) {
779430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      RefNameRange = clang_getCursorReferenceNameRange(Cursor,
780430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                   CXNameRange_WantQualifier
781430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantTemplateArgs,
782430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                       RefNameRangeNr);
783430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (clang_equalRanges(clang_getNullRange(), RefNameRange))
784430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        break;
785430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (!clang_equalRanges(CursorExtent, RefNameRange))
786430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        PrintRange(RefNameRange, "RefName");
787430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    }
788aa0cd85838f2a024e589ea4e8c2094130065af21Dmitri Gribenko
789e4330a302ac20b41b9800267ebd4b5b01f8553f8Dmitri Gribenko    PrintCursorComments(Cursor, ValidationData);
790699a07d8a0b1579c5178b3baf4bcf9edb6b38108Steve Naroff  }
791af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff}
79289922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff
793e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekstatic const char* GetCursorSource(CXCursor Cursor) {
7941db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  CXSourceLocation Loc = clang_getCursorLocation(Cursor);
79574844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  CXString source;
7961db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  CXFile file;
797b4efaa0a14dd2382aa028c03283b5a7f5345e24dArgyrios Kyrtzidis  clang_getExpansionLocation(Loc, &file, 0, 0, 0);
7981db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  source = clang_getFileName(file);
799e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (!clang_getCString(source)) {
80074844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
80174844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    return "<invalid loc>";
80274844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  }
80374844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  else {
804e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    const char *b = basename(clang_getCString(source));
80574844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
80674844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    return b;
80774844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  }
8089298cfc7475c48fa42b168c37f628663d65ddde7Ted Kremenek}
8099298cfc7475c48fa42b168c37f628663d65ddde7Ted Kremenek
8100d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
811ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/* Callbacks.                                                                 */
812ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/******************************************************************************/
813ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek
814ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenektypedef void (*PostVisitTU)(CXTranslationUnit);
815ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek
816a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregorvoid PrintDiagnostic(CXDiagnostic Diagnostic) {
817a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  FILE *out = stderr;
8185352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  CXFile file;
819274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  CXString Msg;
8200a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  unsigned display_opts = CXDiagnostic_DisplaySourceLocation
821aa5f135f8db82b5e5fb1640fd51f8078e0b2d82dDouglas Gregor    | CXDiagnostic_DisplayColumn | CXDiagnostic_DisplaySourceRanges
822aa5f135f8db82b5e5fb1640fd51f8078e0b2d82dDouglas Gregor    | CXDiagnostic_DisplayOption;
8230a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  unsigned i, num_fixits;
824f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
8250a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (clang_getDiagnosticSeverity(Diagnostic) == CXDiagnostic_Ignored)
8265352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return;
827e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
828274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  Msg = clang_formatDiagnostic(Diagnostic, display_opts);
829274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  fprintf(stderr, "%s\n", clang_getCString(Msg));
830274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  clang_disposeString(Msg);
831f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
832a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  clang_getSpellingLocation(clang_getDiagnosticLocation(Diagnostic),
833a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                            &file, 0, 0, 0);
8340a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (!file)
8350a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    return;
836e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
8370a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  num_fixits = clang_getDiagnosticNumFixIts(Diagnostic);
8383739b32a01b9778176757425e62252c288b85383Ted Kremenek  fprintf(stderr, "Number FIX-ITs = %d\n", num_fixits);
8390a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  for (i = 0; i != num_fixits; ++i) {
840473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXSourceRange range;
841473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXString insertion_text = clang_getDiagnosticFixIt(Diagnostic, i, &range);
842473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXSourceLocation start = clang_getRangeStart(range);
843473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXSourceLocation end = clang_getRangeEnd(range);
844473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    unsigned start_line, start_column, end_line, end_column;
845473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXFile start_file, end_file;
846a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(start, &start_file, &start_line,
847a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              &start_column, 0);
848a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(end, &end_file, &end_line, &end_column, 0);
849473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    if (clang_equalLocations(start, end)) {
850473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Insertion. */
851473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      if (start_file == file)
8520a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Insert \"%s\" at %d:%d\n",
853473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor                clang_getCString(insertion_text), start_line, start_column);
854473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    } else if (strcmp(clang_getCString(insertion_text), "") == 0) {
855473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Removal. */
8560a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      if (start_file == file && end_file == file) {
8570a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Remove ");
8580a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        PrintExtent(out, start_line, start_column, end_line, end_column);
8590a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "\n");
860436f3f0400c633251e4071f81358c47bab964adfDouglas Gregor      }
861473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    } else {
862473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Replacement. */
8630a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      if (start_file == end_file) {
8640a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Replace ");
8650a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        PrintExtent(out, start_line, start_column, end_line, end_column);
866473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor        fprintf(out, " with \"%s\"\n", clang_getCString(insertion_text));
86751c6d384551674facc19f745ecf6e289d28dc55fDouglas Gregor      }
8680a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      break;
8690a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    }
870473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    clang_disposeString(insertion_text);
87151c6d384551674facc19f745ecf6e289d28dc55fDouglas Gregor  }
8725352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor}
8735352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor
8747473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenekvoid PrintDiagnosticSet(CXDiagnosticSet Set) {
8757473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  int i = 0, n = clang_getNumDiagnosticsInSet(Set);
8767473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  for ( ; i != n ; ++i) {
8777473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    CXDiagnostic Diag = clang_getDiagnosticInSet(Set, i);
8787473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    CXDiagnosticSet ChildDiags = clang_getChildDiagnostics(Diag);
879a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    PrintDiagnostic(Diag);
8807473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    if (ChildDiags)
8817473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek      PrintDiagnosticSet(ChildDiags);
8827473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  }
8837473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek}
8847473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek
8857473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenekvoid PrintDiagnostics(CXTranslationUnit TU) {
8867473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  CXDiagnosticSet TUSet = clang_getDiagnosticSetFromTU(TU);
8877473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  PrintDiagnosticSet(TUSet);
8887473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  clang_disposeDiagnosticSet(TUSet);
889a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor}
890a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor
89159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenekvoid PrintMemoryUsage(CXTranslationUnit TU) {
892b22732365ca26fbccacc4bdd32e875342f571e94Matt Beaumont-Gay  unsigned long total = 0;
8934e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  unsigned i = 0;
894f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  CXTUResourceUsage usage = clang_getCXTUResourceUsage(TU);
8953c683367073e2d98a9046060f9bc7db872a1c63dFrancois Pichet  fprintf(stderr, "Memory usage:\n");
8964e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  for (i = 0 ; i != usage.numEntries; ++i) {
897f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    const char *name = clang_getTUResourceUsageName(usage.entries[i].kind);
89859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    unsigned long amount = usage.entries[i].amount;
89959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    total += amount;
9004e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    fprintf(stderr, "  %s : %ld bytes (%f MBytes)\n", name, amount,
90159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek            ((double) amount)/(1024*1024));
90259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  }
9034e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  fprintf(stderr, "  TOTAL = %ld bytes (%f MBytes)\n", total,
90459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek          ((double) total)/(1024*1024));
905f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  clang_disposeCXTUResourceUsage(usage);
90659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
90759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
908ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/******************************************************************************/
909e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor/* Logic for testing traversal.                                               */
9100d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
9110d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
912a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregorstatic void PrintCursorExtent(CXCursor C) {
913a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor  CXSourceRange extent = clang_getCursorExtent(C);
914430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  PrintRange(extent, "Extent");
915fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek}
916fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek
917f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko/* Data used by the visitors. */
918f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenkotypedef struct {
919e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  CXTranslationUnit TU;
920e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  enum CXCursorKind *Filter;
921f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  CommentXMLValidationData ValidationData;
922e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor} VisitorData;
923fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek
924625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
925e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekenum CXChildVisitResult FilteredPrintingVisitor(CXCursor Cursor,
926e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                CXCursor Parent,
927e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                CXClientData ClientData) {
928e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
929e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  if (!Data->Filter || (Cursor.kind == *(enum CXCursorKind *)Data->Filter)) {
93098258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    CXSourceLocation Loc = clang_getCursorLocation(Cursor);
9311db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    unsigned line, column;
932a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(Loc, 0, &line, &column, 0);
933fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek    printf("// %s: %s:%d:%d: ", FileCheckPrefix,
9341db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor           GetCursorSource(Cursor), line, column);
935f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(Cursor, &Data->ValidationData);
936a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    PrintCursorExtent(Cursor);
937e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("\n");
938e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Recurse;
939625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  }
940e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
941e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
942625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar}
943c857ea4d22e1e6dd9ede1f0e84d48b157c6924fdSteve Naroff
944e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekstatic enum CXChildVisitResult FunctionScanVisitor(CXCursor Cursor,
945e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                   CXCursor Parent,
946e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                   CXClientData ClientData) {
947625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  const char *startBuf, *endBuf;
948625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  unsigned startLine, startColumn, endLine, endColumn, curLine, curColumn;
949625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXCursor Ref;
950e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
951625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
952b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (Cursor.kind != CXCursor_FunctionDecl ||
953b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      !clang_isCursorDefinition(Cursor))
954e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Continue;
955625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
956625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  clang_getDefinitionSpellingAndExtent(Cursor, &startBuf, &endBuf,
957625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar                                       &startLine, &startColumn,
958625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar                                       &endLine, &endColumn);
959625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  /* Probe the entire body, looking for both decls and refs. */
960625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  curLine = startLine;
961625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  curColumn = startColumn;
962625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
963625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  while (startBuf < endBuf) {
96498258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    CXSourceLocation Loc;
9651db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    CXFile file;
96674844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    CXString source;
967e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
968625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    if (*startBuf == '\n') {
969625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      startBuf++;
970625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curLine++;
971625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curColumn = 1;
972625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    } else if (*startBuf != '\t')
973625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curColumn++;
974e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
97598258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    Loc = clang_getCursorLocation(Cursor);
976a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(Loc, &file, 0, 0, 0);
977e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
9781db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    source = clang_getFileName(file);
979e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    if (clang_getCString(source)) {
980b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor      CXSourceLocation RefLoc
981b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor        = clang_getLocation(Data->TU, file, curLine, curColumn);
982b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor      Ref = clang_getCursor(Data->TU, RefLoc);
98398258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      if (Ref.kind == CXCursor_NoDeclFound) {
98498258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        /* Nothing found here; that's fine. */
98598258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      } else if (Ref.kind != CXCursor_FunctionDecl) {
98698258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        printf("// %s: %s:%d:%d: ", FileCheckPrefix, GetCursorSource(Ref),
98798258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor               curLine, curColumn);
988f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCursor(Ref, &Data->ValidationData);
98998258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        printf("\n");
99098258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      }
9914ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff    }
99274844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
993625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    startBuf++;
9942d4d629d8a0de5112c7ae9d05c03ddbf6dcd956aSteve Naroff  }
995e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
996e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
99789922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff}
99850398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
9997d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
10007d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/* USR testing.                                                               */
10017d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
10027d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
1003e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregorenum CXChildVisitResult USRVisitor(CXCursor C, CXCursor parent,
1004e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                   CXClientData ClientData) {
1005e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
1006e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  if (!Data->Filter || (C.kind == *(enum CXCursorKind *)Data->Filter)) {
1007cf84aa46def41cccf4dd4c51cd0543b70c11e4ebTed Kremenek    CXString USR = clang_getCursorUSR(C);
1008e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    const char *cstr = clang_getCString(USR);
1009e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    if (!cstr || cstr[0] == '\0') {
10107d40562f83552b7295411e10ee887d8d55470679Ted Kremenek      clang_disposeString(USR);
1011e74ef1289d5fff0a6ea573198bf354fa8cd84d51Ted Kremenek      return CXChildVisit_Recurse;
10127d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    }
1013e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    printf("// %s: %s %s", FileCheckPrefix, GetCursorSource(C), cstr);
1014e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek
1015a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    PrintCursorExtent(C);
10167d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    printf("\n");
10177d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    clang_disposeString(USR);
1018e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1019e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Recurse;
1020e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
1021e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1022e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
10237d40562f83552b7295411e10ee887d8d55470679Ted Kremenek}
10247d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
10257d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
102616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/* Inclusion stack testing.                                                   */
102716b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/******************************************************************************/
102816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
102916b55a71695a33c094383295cc7b7a2080e098daTed Kremenekvoid InclusionVisitor(CXFile includedFile, CXSourceLocation *includeStack,
103016b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                      unsigned includeStackLen, CXClientData data) {
1031e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
103216b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  unsigned i;
103374844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  CXString fname;
103474844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek
103574844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  fname = clang_getFileName(includedFile);
1036e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  printf("file: %s\nincluded by:\n", clang_getCString(fname));
103774844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  clang_disposeString(fname);
1038e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
103916b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  for (i = 0; i < includeStackLen; ++i) {
104016b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    CXFile includingFile;
104116b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    unsigned line, column;
1042a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(includeStack[i], &includingFile, &line,
1043a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              &column, 0);
104474844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    fname = clang_getFileName(includingFile);
1045e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("  %s:%d:%d\n", clang_getCString(fname), line, column);
104674844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(fname);
104716b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  }
104816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  printf("\n");
104916b55a71695a33c094383295cc7b7a2080e098daTed Kremenek}
105016b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
105116b55a71695a33c094383295cc7b7a2080e098daTed Kremenekvoid PrintInclusionStack(CXTranslationUnit TU) {
1052e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  clang_getInclusions(TU, InclusionVisitor, NULL);
105316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek}
105416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
105516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/******************************************************************************/
10563bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/* Linkage testing.                                                           */
10573bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/******************************************************************************/
10583bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
10593bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenekstatic enum CXChildVisitResult PrintLinkage(CXCursor cursor, CXCursor p,
10603bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek                                            CXClientData d) {
10613bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  const char *linkage = 0;
10623bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
10633bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  if (clang_isInvalid(clang_getCursorKind(cursor)))
10643bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    return CXChildVisit_Recurse;
10653bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
10663bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  switch (clang_getCursorLinkage(cursor)) {
10673bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    case CXLinkage_Invalid: break;
1068c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_NoLinkage: linkage = "NoLinkage"; break;
1069c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_Internal: linkage = "Internal"; break;
1070c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_UniqueExternal: linkage = "UniqueExternal"; break;
1071c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_External: linkage = "External"; break;
10723bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  }
10733bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
10743bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  if (linkage) {
1075f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(cursor, NULL);
10763bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    printf("linkage=%s\n", linkage);
10773bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  }
10783bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
10793bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  return CXChildVisit_Recurse;
10803bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek}
10813bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
10823bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/******************************************************************************/
10838e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/* Typekind testing.                                                          */
10848e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/******************************************************************************/
10858e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
10868e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenekstatic enum CXChildVisitResult PrintTypeKind(CXCursor cursor, CXCursor p,
10878e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek                                             CXClientData d) {
10888e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  if (!clang_isInvalid(clang_getCursorKind(cursor))) {
10898e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    CXType T = clang_getCursorType(cursor);
10908e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    CXString S = clang_getTypeKindSpelling(T.kind);
1091f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(cursor, NULL);
10928e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    printf(" typekind=%s", clang_getCString(S));
1093e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isConstQualifiedType(T))
1094e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" const");
1095e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isVolatileQualifiedType(T))
1096e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" volatile");
1097e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isRestrictQualifiedType(T))
1098e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" restrict");
10998e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    clang_disposeString(S);
1100e1403d2dcb06fc306e186c787a7bd0ca30e06edeBenjamin Kramer    /* Print the canonical type if it is different. */
110104c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    {
110204c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      CXType CT = clang_getCanonicalType(T);
110304c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      if (!clang_equalTypes(T, CT)) {
110404c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        CXString CS = clang_getTypeKindSpelling(CT.kind);
110504c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        printf(" [canonical=%s]", clang_getCString(CS));
110604c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        clang_disposeString(CS);
110704c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      }
110804c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    }
1109e1403d2dcb06fc306e186c787a7bd0ca30e06edeBenjamin Kramer    /* Print the return type if it exists. */
111004c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    {
11119a140845438c2fc31e7d48a6dedbc695f4c83c68Ted Kremenek      CXType RT = clang_getCursorResultType(cursor);
111204c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      if (RT.kind != CXType_Invalid) {
111304c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        CXString RS = clang_getTypeKindSpelling(RT.kind);
111404c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        printf(" [result=%s]", clang_getCString(RS));
111504c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        clang_disposeString(RS);
111604c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      }
111704c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    }
1118d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    /* Print the argument types if they exist. */
1119d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    {
1120d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis      int numArgs = clang_Cursor_getNumArguments(cursor);
1121d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis      if (numArgs != -1 && numArgs != 0) {
112247f1165c92bc4104e314223ed9ad251e914687c1Argyrios Kyrtzidis        int i;
1123d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        printf(" [args=");
112447f1165c92bc4104e314223ed9ad251e914687c1Argyrios Kyrtzidis        for (i = 0; i < numArgs; ++i) {
1125d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          CXType T = clang_getCursorType(clang_Cursor_getArgument(cursor, i));
1126d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          if (T.kind != CXType_Invalid) {
1127d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis            CXString S = clang_getTypeKindSpelling(T.kind);
1128d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis            printf(" %s", clang_getCString(S));
1129d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis            clang_disposeString(S);
1130d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          }
1131d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        }
1132d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        printf("]");
1133d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis      }
1134d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    }
11353ce9e7d270e7df86c09c8126b4412d55be7c123bTed Kremenek    /* Print if this is a non-POD type. */
11363ce9e7d270e7df86c09c8126b4412d55be7c123bTed Kremenek    printf(" [isPOD=%d]", clang_isPODType(T));
113704c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek
11388e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    printf("\n");
11398e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  }
11408e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  return CXChildVisit_Recurse;
11418e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek}
11428e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
11431eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko/******************************************************************************/
11441eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko/* Bitwidth testing.                                                          */
11451eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko/******************************************************************************/
11461eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko
11471eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenkostatic enum CXChildVisitResult PrintBitWidth(CXCursor cursor, CXCursor p,
11481eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko                                             CXClientData d) {
114902c1b863440b067f3d674bde237146ef54c17875NAKAMURA Takumi  int Bitwidth;
11501eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  if (clang_getCursorKind(cursor) != CXCursor_FieldDecl)
11511eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    return CXChildVisit_Recurse;
11521eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko
115302c1b863440b067f3d674bde237146ef54c17875NAKAMURA Takumi  Bitwidth = clang_getFieldDeclBitWidth(cursor);
11541eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  if (Bitwidth >= 0) {
11551eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    PrintCursor(cursor, NULL);
11561eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    printf(" bitwidth=%d\n", Bitwidth);
11571eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  }
11581eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko
11591eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  return CXChildVisit_Recurse;
11601eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko}
11618e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
11628e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/******************************************************************************/
11637d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/* Loading ASTs/source.                                                       */
11647d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
11657d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
1166625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbarstatic int perform_test_load(CXIndex Idx, CXTranslationUnit TU,
116798271567b6f10be85bcc75d0bbb67416c8cfb83aTed Kremenek                             const char *filter, const char *prefix,
1168ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             CXCursorVisitor Visitor,
1169f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             PostVisitTU PV,
1170f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             const char *CommentSchemaFile) {
1171e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1172fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek  if (prefix)
1173e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    FileCheckPrefix = prefix;
1174e3ee02a324559829142f387f75daf3da118442beTed Kremenek
1175e3ee02a324559829142f387f75daf3da118442beTed Kremenek  if (Visitor) {
1176e3ee02a324559829142f387f75daf3da118442beTed Kremenek    enum CXCursorKind K = CXCursor_NotImplemented;
1177e3ee02a324559829142f387f75daf3da118442beTed Kremenek    enum CXCursorKind *ck = &K;
1178e3ee02a324559829142f387f75daf3da118442beTed Kremenek    VisitorData Data;
1179e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1180e3ee02a324559829142f387f75daf3da118442beTed Kremenek    /* Perform some simple filtering. */
1181e3ee02a324559829142f387f75daf3da118442beTed Kremenek    if (!strcmp(filter, "all") || !strcmp(filter, "local")) ck = NULL;
1182358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    else if (!strcmp(filter, "all-display") ||
1183358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor             !strcmp(filter, "local-display")) {
1184358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      ck = NULL;
1185358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      want_display_name = 1;
1186358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
1187b1ffee6e574d57b79ce2defd1b57052649221c03Daniel Dunbar    else if (!strcmp(filter, "none")) K = (enum CXCursorKind) ~0;
1188e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "category")) K = CXCursor_ObjCCategoryDecl;
1189e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "interface")) K = CXCursor_ObjCInterfaceDecl;
1190e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "protocol")) K = CXCursor_ObjCProtocolDecl;
1191e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "function")) K = CXCursor_FunctionDecl;
1192e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "typedef")) K = CXCursor_TypedefDecl;
1193e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "scan-function")) Visitor = FunctionScanVisitor;
1194e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else {
1195e3ee02a324559829142f387f75daf3da118442beTed Kremenek      fprintf(stderr, "Unknown filter for -test-load-tu: %s\n", filter);
1196e3ee02a324559829142f387f75daf3da118442beTed Kremenek      return 1;
1197e3ee02a324559829142f387f75daf3da118442beTed Kremenek    }
1198e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1199e3ee02a324559829142f387f75daf3da118442beTed Kremenek    Data.TU = TU;
1200e3ee02a324559829142f387f75daf3da118442beTed Kremenek    Data.Filter = ck;
1201f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    Data.ValidationData.CommentSchemaFile = CommentSchemaFile;
1202f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
1203f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    Data.ValidationData.RNGParser = NULL;
1204f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    Data.ValidationData.Schema = NULL;
1205f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
1206e3ee02a324559829142f387f75daf3da118442beTed Kremenek    clang_visitChildren(clang_getTranslationUnitCursor(TU), Visitor, &Data);
1207e3ee02a324559829142f387f75daf3da118442beTed Kremenek  }
1208e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1209ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek  if (PV)
1210ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek    PV(TU);
1211e3ee02a324559829142f387f75daf3da118442beTed Kremenek
1212a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
121316ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
121416ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis    clang_disposeTranslationUnit(TU);
121516ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis    return -1;
121616ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis  }
121716ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis
12180d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek  clang_disposeTranslationUnit(TU);
12190d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek  return 0;
12200d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek}
12210d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
1222fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenekint perform_test_load_tu(const char *file, const char *filter,
1223ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                         const char *prefix, CXCursorVisitor Visitor,
1224ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                         PostVisitTU PV) {
1225625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXIndex Idx;
1226625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXTranslationUnit TU;
1227020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  int result;
1228e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  Idx = clang_createIndex(/* excludeDeclsFromPCH */
12290a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                          !strcmp(filter, "local") ? 1 : 0,
12300a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                          /* displayDiagnosics=*/1);
1231e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1232020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  if (!CreateTranslationUnit(Idx, file, &TU)) {
1233020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
1234625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    return 1;
1235020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  }
1236625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1237f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, prefix, Visitor, PV, NULL);
1238020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  clang_disposeIndex(Idx);
1239020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  return result;
1240625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar}
1241625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1242ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenekint perform_test_load_source(int argc, const char **argv,
1243ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             const char *filter, CXCursorVisitor Visitor,
1244ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             PostVisitTU PV) {
1245ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  CXIndex Idx;
1246ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  CXTranslationUnit TU;
1247f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  const char *CommentSchemaFile;
12484db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
12494db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int num_unsaved_files = 0;
12504db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int result;
1251abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1252ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  Idx = clang_createIndex(/* excludeDeclsFromPCH */
1253358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                          (!strcmp(filter, "local") ||
1254358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                           !strcmp(filter, "local-display"))? 1 : 0,
12554814fb560322daf009fda6226cffccb10bd44620Douglas Gregor                          /* displayDiagnosics=*/0);
1256ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1257f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  if ((CommentSchemaFile = parse_comments_schema(argc, argv))) {
1258f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    argc--;
1259f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    argv++;
1260f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  }
1261f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
1262020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
1263020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
12644db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    return -1;
1265020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  }
12664db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
1267dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor  TU = clang_parseTranslationUnit(Idx, 0,
1268dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  argv + num_unsaved_files,
1269dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  argc - num_unsaved_files,
1270dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  unsaved_files, num_unsaved_files,
1271dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  getDefaultParsingOptions());
1272ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  if (!TU) {
1273ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar    fprintf(stderr, "Unable to load translation unit!\n");
1274abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
1275020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
1276ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar    return 1;
1277ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  }
1278ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1279f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, NULL, Visitor, PV,
1280f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko                             CommentSchemaFile);
12814db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1282020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  clang_disposeIndex(Idx);
12834db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  return result;
1284ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar}
1285ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1286abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregorint perform_test_reparse_source(int argc, const char **argv, int trials,
1287abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                const char *filter, CXCursorVisitor Visitor,
1288abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                PostVisitTU PV) {
1289abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  CXIndex Idx;
1290abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  CXTranslationUnit TU;
1291abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
1292abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int num_unsaved_files = 0;
1293abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int result;
1294abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int trial;
129540098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  int remap_after_trial = 0;
129640098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  char *endptr = 0;
1297abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1298abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  Idx = clang_createIndex(/* excludeDeclsFromPCH */
1299abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                          !strcmp(filter, "local") ? 1 : 0,
13001aa27307c462baaa9e5fda14ff6797dd39fe8b84Douglas Gregor                          /* displayDiagnosics=*/0);
1301abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1302abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
1303abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    clang_disposeIndex(Idx);
1304abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    return -1;
1305abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1306abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1307c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar  /* Load the initial translation unit -- we do this without honoring remapped
1308c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar   * files, so that we have a way to test results after changing the source. */
130944c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  TU = clang_parseTranslationUnit(Idx, 0,
131044c181aec37789f25f6c15543c164416f72e562aDouglas Gregor                                  argv + num_unsaved_files,
131144c181aec37789f25f6c15543c164416f72e562aDouglas Gregor                                  argc - num_unsaved_files,
1312c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar                                  0, 0, getDefaultParsingOptions());
1313abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  if (!TU) {
1314abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
1315abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
1316abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    clang_disposeIndex(Idx);
1317abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    return 1;
1318abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1319abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1320bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
1321bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis    return -1;
1322bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
132340098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  if (getenv("CINDEXTEST_REMAP_AFTER_TRIAL")) {
132440098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis    remap_after_trial =
132540098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis        strtol(getenv("CINDEXTEST_REMAP_AFTER_TRIAL"), &endptr, 10);
132640098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  }
132740098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis
1328abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  for (trial = 0; trial < trials; ++trial) {
132940098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis    if (clang_reparseTranslationUnit(TU,
133040098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis                             trial >= remap_after_trial ? num_unsaved_files : 0,
133140098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis                             trial >= remap_after_trial ? unsaved_files : 0,
1332e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregor                                     clang_defaultReparseOptions(TU))) {
1333c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar      fprintf(stderr, "Unable to reparse translation unit!\n");
1334abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      clang_disposeTranslationUnit(TU);
1335abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      free_remapped_files(unsaved_files, num_unsaved_files);
1336abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      clang_disposeIndex(Idx);
1337abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      return -1;
1338abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    }
1339bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
1340bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
1341bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis      return -1;
1342abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1343abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1344f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  result = perform_test_load(Idx, TU, filter, NULL, Visitor, PV, NULL);
1345bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
1346abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1347abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  clang_disposeIndex(Idx);
1348abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  return result;
1349abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor}
1350abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
13510d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
13521c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/* Logic for testing clang_getCursor().                                       */
13531c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/******************************************************************************/
13541c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
1355dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregorstatic void print_cursor_file_scan(CXTranslationUnit TU, CXCursor cursor,
13561c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek                                   unsigned start_line, unsigned start_col,
13571d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                                   unsigned end_line, unsigned end_col,
13581d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                                   const char *prefix) {
13599096a20fc0fd799911b738ce7a9134dbacaf64b7Ted Kremenek  printf("// %s: ", FileCheckPrefix);
13601d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek  if (prefix)
13611d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    printf("-%s", prefix);
136251b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  PrintExtent(stdout, start_line, start_col, end_line, end_col);
136351b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  printf(" ");
1364f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(cursor, NULL);
13651c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  printf("\n");
13661c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
13671c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
13681d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenekstatic int perform_file_scan(const char *ast_file, const char *source_file,
13691d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                             const char *prefix) {
13701c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  CXIndex Idx;
13711c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  CXTranslationUnit TU;
13721c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  FILE *fp;
13732389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  CXCursor prevCursor = clang_getNullCursor();
1374b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  CXFile file;
13752389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  unsigned line = 1, col = 1;
13768f0bf81c0f3ab31881cc7db7914e6978bb2a19afDaniel Dunbar  unsigned start_line = 1, start_col = 1;
1377e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
13780a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
13790a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                                /* displayDiagnosics=*/1))) {
13801c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Could not create Index\n");
13811c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
13821c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1383e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
13841c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  if (!CreateTranslationUnit(Idx, ast_file, &TU))
13851c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
1386e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
13871c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  if ((fp = fopen(source_file, "r")) == NULL) {
13881c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Could not open '%s'\n", source_file);
13891c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
13901c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1391e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1392b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  file = clang_getFile(TU, source_file);
13932389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  for (;;) {
13942389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    CXCursor cursor;
13952389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    int c = fgetc(fp);
13962389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar
13972389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if (c == '\n') {
13982389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      ++line;
13992389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      col = 1;
14002389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    } else
14012389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      ++col;
14022389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar
14032389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    /* Check the cursor at this position, and dump the previous one if we have
14042389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar     * found something new.
14052389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar     */
14062389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    cursor = clang_getCursor(TU, clang_getLocation(TU, file, line, col));
14072389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if ((c == EOF || !clang_equalCursors(cursor, prevCursor)) &&
14082389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar        prevCursor.kind != CXCursor_InvalidFile) {
1409dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor      print_cursor_file_scan(TU, prevCursor, start_line, start_col,
1410d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar                             line, col, prefix);
14112389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      start_line = line;
14122389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      start_col = col;
1413a9933b98399f656653a0876fc39e5b9093efb732Benjamin Kramer    }
14142389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if (c == EOF)
14152389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      break;
1416a9933b98399f656653a0876fc39e5b9093efb732Benjamin Kramer
14172389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    prevCursor = cursor;
14181c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1419e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
14201c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  fclose(fp);
14214f5e21e24fb9e6ec473a13f83b5c9a2c41501a70Douglas Gregor  clang_disposeTranslationUnit(TU);
14224f5e21e24fb9e6ec473a13f83b5c9a2c41501a70Douglas Gregor  clang_disposeIndex(Idx);
14231c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  return 0;
14241c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
14251c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
14261c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/******************************************************************************/
142732be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor/* Logic for testing clang code completion.                                   */
14280d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
14290d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
14300c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor/* Parse file:line:column from the input string. Returns 0 on success, non-zero
14310c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor   on failure. If successful, the pointer *filename will contain newly-allocated
14320c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor   memory (that will be owned by the caller) to store the file name. */
1433e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekint parse_file_line_column(const char *input, char **filename, unsigned *line,
1434fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                           unsigned *column, unsigned *second_line,
1435fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                           unsigned *second_column) {
143688d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor  /* Find the second colon. */
1437fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  const char *last_colon = strrchr(input, ':');
1438fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned values[4], i;
1439fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned num_values = (second_line && second_column)? 4 : 2;
1440fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
14410c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  char *endptr = 0;
1442fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!last_colon || last_colon == input) {
1443fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (num_values == 4)
1444fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      fprintf(stderr, "could not parse filename:line:column:line:column in "
1445fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              "'%s'\n", input);
1446fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    else
1447fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      fprintf(stderr, "could not parse filename:line:column in '%s'\n", input);
14480c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor    return 1;
14490c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
14500c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1451fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  for (i = 0; i != num_values; ++i) {
1452fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *prev_colon;
145388d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor
1454fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    /* Parse the next line or column. */
1455fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    values[num_values - i - 1] = strtol(last_colon + 1, &endptr, 10);
1456fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (*endptr != 0 && *endptr != ':') {
1457e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr, "could not parse %s in '%s'\n",
1458fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              (i % 2 ? "column" : "line"), input);
1459fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      return 1;
1460fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
1461e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1462fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (i + 1 == num_values)
1463fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      break;
146488d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor
1465fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    /* Find the previous colon. */
1466fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    prev_colon = last_colon - 1;
1467fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    while (prev_colon != input && *prev_colon != ':')
1468fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      --prev_colon;
1469fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (prev_colon == input) {
1470e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr, "could not parse %s in '%s'\n",
1471fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              (i % 2 == 0? "column" : "line"), input);
1472e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return 1;
1473fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
1474fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
1475fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    last_colon = prev_colon;
14760c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
1477fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
1478fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *line = values[0];
1479fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *column = values[1];
1480e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1481fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (second_line && second_column) {
1482fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *second_line = values[2];
1483fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *second_column = values[3];
1484fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
1485fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
148688d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor  /* Copy the file name. */
1487fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *filename = (char*)malloc(last_colon - input + 1);
1488fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  memcpy(*filename, input, last_colon - input);
1489fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  (*filename)[last_colon - input] = 0;
14900c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  return 0;
14910c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
14920c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
14930c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregorconst char *
14940c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregorclang_getCompletionChunkKindSpelling(enum CXCompletionChunkKind Kind) {
14950c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  switch (Kind) {
14960c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Optional: return "Optional";
14970c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_TypedText: return "TypedText";
14980c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Text: return "Text";
14990c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Placeholder: return "Placeholder";
15000c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Informative: return "Informative";
15010c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_CurrentParameter: return "CurrentParameter";
15020c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftParen: return "LeftParen";
15030c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightParen: return "RightParen";
15040c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftBracket: return "LeftBracket";
15050c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightBracket: return "RightBracket";
15060c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftBrace: return "LeftBrace";
15070c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightBrace: return "RightBrace";
15080c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftAngle: return "LeftAngle";
15090c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightAngle: return "RightAngle";
15100c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Comma: return "Comma";
1511ff5ce6eefc7c253ef6edf4d4bfc996fdd82d09aaDouglas Gregor  case CXCompletionChunk_ResultType: return "ResultType";
151201dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_Colon: return "Colon";
151301dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_SemiColon: return "SemiColon";
151401dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_Equal: return "Equal";
151501dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_HorizontalSpace: return "HorizontalSpace";
151601dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_VerticalSpace: return "VerticalSpace";
15170c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
1518e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
15190c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  return "Unknown";
15200c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
15210c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1522dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidisstatic int checkForErrors(CXTranslationUnit TU) {
1523dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  unsigned Num, i;
1524dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  CXDiagnostic Diag;
1525dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  CXString DiagStr;
1526dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1527dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (!getenv("CINDEXTEST_FAILONERROR"))
1528dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return 0;
1529dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1530dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  Num = clang_getNumDiagnostics(TU);
1531dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  for (i = 0; i != Num; ++i) {
1532dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    Diag = clang_getDiagnostic(TU, i);
1533dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (clang_getDiagnosticSeverity(Diag) >= CXDiagnostic_Error) {
1534dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      DiagStr = clang_formatDiagnostic(Diag,
1535dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis                                       clang_defaultDiagnosticDisplayOptions());
1536dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      fprintf(stderr, "%s\n", clang_getCString(DiagStr));
1537dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      clang_disposeString(DiagStr);
1538dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      clang_disposeDiagnostic(Diag);
1539dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
1540dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    }
1541dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    clang_disposeDiagnostic(Diag);
1542dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
1543dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1544dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  return 0;
1545dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis}
1546dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
15473ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregorvoid print_completion_string(CXCompletionString completion_string, FILE *file) {
1548f8297f1512d29c88f20bf2901f04cc04182a3eeeDaniel Dunbar  int I, N;
1549e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
15503ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  N = clang_getNumCompletionChunks(completion_string);
15510c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  for (I = 0; I != N; ++I) {
15522ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    CXString text;
15532ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    const char *cstr;
15540c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor    enum CXCompletionChunkKind Kind
15553ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      = clang_getCompletionChunkKind(completion_string, I);
1556e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
15573ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor    if (Kind == CXCompletionChunk_Optional) {
15583ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      fprintf(file, "{Optional ");
15593ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      print_completion_string(
1560e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                clang_getCompletionChunkCompletionString(completion_string, I),
15613ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor                              file);
15623ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      fprintf(file, "}");
15633ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      continue;
15645a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor    }
15655a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor
15665a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor    if (Kind == CXCompletionChunk_VerticalSpace) {
15675a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor      fprintf(file, "{VerticalSpace  }");
15685a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor      continue;
15693ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor    }
1570e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1571d5a2089663d81faee0de031deabc418fa53ecf3bDouglas Gregor    text = clang_getCompletionChunkText(completion_string, I);
15722ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    cstr = clang_getCString(text);
1573e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(file, "{%s %s}",
15740c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor            clang_getCompletionChunkKindSpelling(Kind),
15752ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek            cstr ? cstr : "");
15762ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    clang_disposeString(text);
15770c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
15782ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek
15793ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor}
15803ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor
15813ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregorvoid print_completion_result(CXCompletionResult *completion_result,
15823ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor                             CXClientData client_data) {
15833ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  FILE *file = (FILE *)client_data;
1584e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  CXString ks = clang_getCursorKindSpelling(completion_result->CursorKind);
15856164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  unsigned annotationCount;
1586ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  enum CXCursorKind ParentKind;
1587ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  CXString ParentName;
1588d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  CXString BriefComment;
1589d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  const char *BriefCommentCString;
1590ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor
1591e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  fprintf(file, "%s:", clang_getCString(ks));
1592e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  clang_disposeString(ks);
1593e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
15943ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  print_completion_string(completion_result->CompletionString, file);
159558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  fprintf(file, " (%u)",
159612e131385e892e3723483a1081a89bcad29c8a84Douglas Gregor          clang_getCompletionPriority(completion_result->CompletionString));
159758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  switch (clang_getCompletionAvailability(completion_result->CompletionString)){
159858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_Available:
159958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
160058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
160158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_Deprecated:
160258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    fprintf(file, " (deprecated)");
160358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
160458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
160558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_NotAvailable:
160658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    fprintf(file, " (unavailable)");
160758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
1608d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen
1609d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen  case CXAvailability_NotAccessible:
1610d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen    fprintf(file, " (inaccessible)");
1611d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen    break;
161258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  }
16136164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen
16146164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  annotationCount = clang_getCompletionNumAnnotations(
16156164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen        completion_result->CompletionString);
16166164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  if (annotationCount) {
16176164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    unsigned i;
16186164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    fprintf(file, " (");
16196164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    for (i = 0; i < annotationCount; ++i) {
16206164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen      if (i != 0)
16216164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen        fprintf(file, ", ");
16226164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen      fprintf(file, "\"%s\"",
16236164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen              clang_getCString(clang_getCompletionAnnotation(
16246164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen                                 completion_result->CompletionString, i)));
16256164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    }
16266164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    fprintf(file, ")");
16276164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  }
16286164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen
1629ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  if (!getenv("CINDEXTEST_NO_COMPLETION_PARENTS")) {
1630ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    ParentName = clang_getCompletionParent(completion_result->CompletionString,
1631ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor                                           &ParentKind);
1632ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    if (ParentKind != CXCursor_NotImplemented) {
1633ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      CXString KindSpelling = clang_getCursorKindSpelling(ParentKind);
1634ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      fprintf(file, " (parent: %s '%s')",
1635ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor              clang_getCString(KindSpelling),
1636ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor              clang_getCString(ParentName));
1637ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      clang_disposeString(KindSpelling);
1638ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    }
1639ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    clang_disposeString(ParentName);
1640ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  }
1641d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko
1642d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  BriefComment = clang_getCompletionBriefComment(
1643d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko                                        completion_result->CompletionString);
1644d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  BriefCommentCString = clang_getCString(BriefComment);
1645d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (BriefCommentCString && *BriefCommentCString != '\0') {
1646d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    fprintf(file, "(brief comment: %s)", BriefCommentCString);
1647d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  }
1648d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  clang_disposeString(BriefComment);
1649ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor
165058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  fprintf(file, "\n");
16510c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
16520c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
16533da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregorvoid print_completion_contexts(unsigned long long contexts, FILE *file) {
16543da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  fprintf(file, "Completion contexts:\n");
16553da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts == CXCompletionContext_Unknown) {
16563da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Unknown\n");
16573da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16583da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_AnyType) {
16593da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Any type\n");
16603da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16613da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_AnyValue) {
16623da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Any value\n");
16633da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16643da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCObjectValue) {
16653da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C object value\n");
16663da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16673da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCSelectorValue) {
16683da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C selector value\n");
16693da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16703da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_CXXClassTypeValue) {
16713da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "C++ class type value\n");
16723da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16733da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_DotMemberAccess) {
16743da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Dot member access\n");
16753da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16763da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ArrowMemberAccess) {
16773da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Arrow member access\n");
16783da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16793da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCPropertyAccess) {
16803da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C property access\n");
16813da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16823da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_EnumTag) {
16833da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Enum tag\n");
16843da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16853da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_UnionTag) {
16863da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Union tag\n");
16873da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16883da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_StructTag) {
16893da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Struct tag\n");
16903da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16913da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ClassTag) {
16923da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Class name\n");
16933da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16943da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_Namespace) {
16953da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Namespace or namespace alias\n");
16963da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
16973da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_NestedNameSpecifier) {
16983da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Nested name specifier\n");
16993da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
17003da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCInterface) {
17013da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C interface\n");
17023da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
17033da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCProtocol) {
17043da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C protocol\n");
17053da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
17063da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCCategory) {
17073da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C category\n");
17083da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
17093da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCInstanceMessage) {
17103da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C instance method\n");
17113da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
17123da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCClassMessage) {
17133da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C class method\n");
17143da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
17153da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCSelectorName) {
17163da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C selector name\n");
17173da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
17183da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_MacroName) {
17193da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Macro name\n");
17203da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
17213da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_NaturalLanguage) {
17223da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Natural language\n");
17233da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
17243da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor}
17253da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
17261e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregorint my_stricmp(const char *s1, const char *s2) {
17271e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  while (*s1 && *s2) {
17286d5552131a4637f9bbe8c93386648e9bbb2c30feNAKAMURA Takumi    int c1 = tolower((unsigned char)*s1), c2 = tolower((unsigned char)*s2);
17291e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    if (c1 < c2)
17301e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      return -1;
17311e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    else if (c1 > c2)
17321e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      return 1;
17331e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
17341e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    ++s1;
17351e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    ++s2;
17361e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  }
17371e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
17381e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  if (*s1)
17391e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    return 1;
17401e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  else if (*s2)
17411e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    return -1;
17421e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  return 0;
17431e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor}
17441e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
17451982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregorint perform_code_completion(int argc, const char **argv, int timing_only) {
17460c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  const char *input = argv[1];
17470c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  char *filename = 0;
17480c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  unsigned line;
17490c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  unsigned column;
1750f8297f1512d29c88f20bf2901f04cc04182a3eeeDaniel Dunbar  CXIndex CIdx;
1751f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  int errorCode;
1752735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
1753735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  int num_unsaved_files = 0;
1754ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  CXCodeCompleteResults *results = 0;
175525d9b00ab0b128d651d993c38726a00cd9969124Dawn Perchik  CXTranslationUnit TU = 0;
175632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  unsigned I, Repeats = 1;
175732be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  unsigned completionOptions = clang_defaultCodeCompleteOptions();
175832be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
175932be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (getenv("CINDEXTEST_CODE_COMPLETE_PATTERNS"))
176032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    completionOptions |= CXCodeComplete_IncludeCodePatterns;
1761d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (getenv("CINDEXTEST_COMPLETION_BRIEF_COMMENTS"))
1762d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    completionOptions |= CXCodeComplete_IncludeBriefComments;
1763df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor
17641982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  if (timing_only)
17651982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    input += strlen("-code-completion-timing=");
17661982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  else
17671982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    input += strlen("-code-completion-at=");
17681982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor
1769e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if ((errorCode = parse_file_line_column(input, &filename, &line, &column,
1770fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                          0, 0)))
1771f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek    return errorCode;
17720c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1773735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  if (parse_remapped_files(argc, argv, 2, &unsaved_files, &num_unsaved_files))
1774735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor    return -1;
1775735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor
177632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  CIdx = clang_createIndex(0, 0);
177732be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
177832be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (getenv("CINDEXTEST_EDITING"))
177932be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    Repeats = 5;
178032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
178132be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  TU = clang_parseTranslationUnit(CIdx, 0,
178232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                  argv + num_unsaved_files + 2,
178332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                  argc - num_unsaved_files - 2,
178432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                  0, 0, getDefaultParsingOptions());
178532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (!TU) {
178632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
178732be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    return 1;
178832be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  }
178908bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor
179008bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor  if (clang_reparseTranslationUnit(TU, 0, 0, clang_defaultReparseOptions(TU))) {
179108bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor    fprintf(stderr, "Unable to reparse translation init!\n");
179208bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor    return 1;
179308bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor  }
179432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
179532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  for (I = 0; I != Repeats; ++I) {
179632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    results = clang_codeCompleteAt(TU, filename, line, column,
179732be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                   unsaved_files, num_unsaved_files,
179832be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                   completionOptions);
179932be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    if (!results) {
180032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor      fprintf(stderr, "Unable to perform code completion!\n");
18012de41c9c99e0e9ae6488d04c08423a5c1190109eDaniel Dunbar      return 1;
18022de41c9c99e0e9ae6488d04c08423a5c1190109eDaniel Dunbar    }
180332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    if (I != Repeats-1)
180432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor      clang_disposeCodeCompleteResults(results);
180532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  }
1806936ea3b590117d2cd73b1b92621d06c4a7edbe60Douglas Gregor
1807ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  if (results) {
1808e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    unsigned i, n = results->NumResults, containerIsIncomplete = 0;
18093da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    unsigned long long contexts;
1810e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    enum CXCursorKind containerKind;
18110a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    CXString objCSelector;
18120a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    const char *selectorString;
18131e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    if (!timing_only) {
18141e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      /* Sort the code-completion results based on the typed text. */
18151e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      clang_sortCodeCompletionResults(results->Results, results->NumResults);
18161e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
18171982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor      for (i = 0; i != n; ++i)
18181982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor        print_completion_result(results->Results + i, stdout);
18191e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    }
1820a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    n = clang_codeCompleteGetNumDiagnostics(results);
1821a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    for (i = 0; i != n; ++i) {
1822a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      CXDiagnostic diag = clang_codeCompleteGetDiagnostic(results, i);
1823a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      PrintDiagnostic(diag);
1824a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      clang_disposeDiagnostic(diag);
1825a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    }
18263da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
18273da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    contexts = clang_codeCompleteGetContexts(results);
18283da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    print_completion_contexts(contexts, stdout);
18293da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
18300a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    containerKind = clang_codeCompleteGetContainerKind(results,
18310a47d69af8bda945352997af3da4687a3356096aDouglas Gregor                                                       &containerIsIncomplete);
1832e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
1833e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    if (containerKind != CXCursor_InvalidCode) {
1834e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      /* We have found a container */
1835e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      CXString containerUSR, containerKindSpelling;
1836e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      containerKindSpelling = clang_getCursorKindSpelling(containerKind);
1837e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      printf("Container Kind: %s\n", clang_getCString(containerKindSpelling));
1838e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      clang_disposeString(containerKindSpelling);
1839e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
1840e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      if (containerIsIncomplete) {
1841e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor        printf("Container is incomplete\n");
1842e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      }
1843e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      else {
1844e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor        printf("Container is complete\n");
1845e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      }
1846e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
1847e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      containerUSR = clang_codeCompleteGetContainerUSR(results);
1848e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      printf("Container USR: %s\n", clang_getCString(containerUSR));
1849e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      clang_disposeString(containerUSR);
1850e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    }
1851e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
18520a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    objCSelector = clang_codeCompleteGetObjCSelector(results);
18530a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    selectorString = clang_getCString(objCSelector);
18540a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    if (selectorString && strlen(selectorString) > 0) {
18550a47d69af8bda945352997af3da4687a3356096aDouglas Gregor      printf("Objective-C selector: %s\n", selectorString);
18560a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    }
18570a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    clang_disposeString(objCSelector);
18580a47d69af8bda945352997af3da4687a3356096aDouglas Gregor
1859ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor    clang_disposeCodeCompleteResults(results);
1860ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  }
1861df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor  clang_disposeTranslationUnit(TU);
18620c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  clang_disposeIndex(CIdx);
18630c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  free(filename);
1864e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1865735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1866735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor
1867f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  return 0;
18680c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
18690c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1870f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregortypedef struct {
1871f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  char *filename;
1872f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned line;
1873f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned column;
1874f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor} CursorSourceLocation;
1875f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor
1876aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic int inspect_cursor_at(int argc, const char **argv) {
1877f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXIndex CIdx;
1878f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  int errorCode;
1879f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
1880f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  int num_unsaved_files = 0;
1881f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXTranslationUnit TU;
1882f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXCursor Cursor;
1883f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CursorSourceLocation *Locations = 0;
1884f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned NumLocations = 0, Loc;
18858e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  unsigned Repeats = 1;
1886bdc4b366e80c125184a3b3c56fa4619cb4ac9e45Douglas Gregor  unsigned I;
18878e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
1888e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  /* Count the number of locations. */
1889f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  while (strstr(argv[NumLocations+1], "-cursor-at=") == argv[NumLocations+1])
1890f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    ++NumLocations;
1891e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1892f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  /* Parse the locations. */
1893f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  assert(NumLocations > 0 && "Unable to count locations?");
1894f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  Locations = (CursorSourceLocation *)malloc(
1895f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                                  NumLocations * sizeof(CursorSourceLocation));
1896f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  for (Loc = 0; Loc < NumLocations; ++Loc) {
1897f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    const char *input = argv[Loc + 1] + strlen("-cursor-at=");
1898e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    if ((errorCode = parse_file_line_column(input, &Locations[Loc].filename,
1899e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                                            &Locations[Loc].line,
1900fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                            &Locations[Loc].column, 0, 0)))
1901f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor      return errorCode;
1902f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
1903e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1904e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (parse_remapped_files(argc, argv, NumLocations + 1, &unsaved_files,
1905f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                           &num_unsaved_files))
1906f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return -1;
1907e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
19088e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  if (getenv("CINDEXTEST_EDITING"))
19098e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    Repeats = 5;
19108e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
19118e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  /* Parse the translation unit. When we're testing clang_getCursor() after
19128e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor     reparsing, don't remap unsaved files until the second parse. */
19138e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  CIdx = clang_createIndex(1, 1);
19148e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  TU = clang_parseTranslationUnit(CIdx, argv[argc - 1],
19158e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                  argv + num_unsaved_files + 1 + NumLocations,
1916f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                                  argc - num_unsaved_files - 2 - NumLocations,
19178e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                  unsaved_files,
19188e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                  Repeats > 1? 0 : num_unsaved_files,
19198e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                  getDefaultParsingOptions());
19208e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
1921f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  if (!TU) {
1922f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    fprintf(stderr, "unable to parse input\n");
1923f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return -1;
1924f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
1925e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1926dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
1927dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return -1;
1928dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1929bdc4b366e80c125184a3b3c56fa4619cb4ac9e45Douglas Gregor  for (I = 0; I != Repeats; ++I) {
19308e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    if (Repeats > 1 &&
19318e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
19328e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                     clang_defaultReparseOptions(TU))) {
19338e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      clang_disposeTranslationUnit(TU);
19348e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      return 1;
19358e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    }
1936dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1937dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
1938dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
19398e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
19408e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    for (Loc = 0; Loc < NumLocations; ++Loc) {
19418e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      CXFile file = clang_getFile(TU, Locations[Loc].filename);
19428e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      if (!file)
19438e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        continue;
19448e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
19458e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      Cursor = clang_getCursor(TU,
19468e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                               clang_getLocation(TU, file, Locations[Loc].line,
19478e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                                 Locations[Loc].column));
1948dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1949dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      if (checkForErrors(TU) != 0)
1950dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        return -1;
1951dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
19528e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      if (I + 1 == Repeats) {
19538fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor        CXCompletionString completionString = clang_getCursorCompletionString(
19548fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor                                                                        Cursor);
195566373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        CXSourceLocation CursorLoc = clang_getCursorLocation(Cursor);
195666373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        CXString Spelling;
195766373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        const char *cspell;
195866373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        unsigned line, column;
195966373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        clang_getSpellingLocation(CursorLoc, 0, &line, &column, 0);
196066373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        printf("%d:%d ", line, column);
1961f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCursor(Cursor, NULL);
196266373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        PrintCursorExtent(Cursor);
196366373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        Spelling = clang_getCursorSpelling(Cursor);
196466373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        cspell = clang_getCString(Spelling);
1965ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis        if (cspell && strlen(cspell) != 0) {
1966ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          unsigned pieceIndex;
1967ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          printf(" Spelling=%s (", cspell);
1968ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          for (pieceIndex = 0; ; ++pieceIndex) {
19696c235bc9e126fcecd16257838aa040fc9c363328Benjamin Kramer            CXSourceRange range =
19706c235bc9e126fcecd16257838aa040fc9c363328Benjamin Kramer              clang_Cursor_getSpellingNameRange(Cursor, pieceIndex, 0);
1971ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis            if (clang_Range_isNull(range))
1972ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis              break;
1973ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis            PrintRange(range, 0);
1974ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          }
1975ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          printf(")");
1976ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis        }
197766373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        clang_disposeString(Spelling);
197834ebe1e1b0779bcea2f277bc6b4e9dd98bf70b7bArgyrios Kyrtzidis        if (clang_Cursor_getObjCSelectorIndex(Cursor) != -1)
197934ebe1e1b0779bcea2f277bc6b4e9dd98bf70b7bArgyrios Kyrtzidis          printf(" Selector index=%d",clang_Cursor_getObjCSelectorIndex(Cursor));
1980f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis        if (clang_Cursor_isDynamicCall(Cursor))
1981f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis          printf(" Dynamic-call");
1982e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis        if (Cursor.kind == CXCursor_ObjCMessageExpr) {
1983e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis          CXType T = clang_Cursor_getReceiverType(Cursor);
1984e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis          CXString S = clang_getTypeKindSpelling(T.kind);
1985e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis          printf(" Receiver-type=%s", clang_getCString(S));
1986e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis          clang_disposeString(S);
1987e4a990f34904eb572c8d6aa1deef19465214359cArgyrios Kyrtzidis        }
1988f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis
19895d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis        {
19905d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis          CXModule mod = clang_Cursor_getModule(Cursor);
19915d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis          CXString name;
19925d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis          unsigned i, numHeaders;
19935d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis          if (mod) {
19945d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            name = clang_Module_getFullName(mod);
19955d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            numHeaders = clang_Module_getNumTopLevelHeaders(mod);
19965d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            printf(" ModuleName=%s Headers(%d):",
19975d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis                   clang_getCString(name), numHeaders);
19985d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            clang_disposeString(name);
19995d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            for (i = 0; i < numHeaders; ++i) {
20005d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis              CXFile file = clang_Module_getTopLevelHeader(mod, i);
20015d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis              CXString filename = clang_getFileName(file);
20025d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis              printf("\n%s", clang_getCString(filename));
20035d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis              clang_disposeString(filename);
20045d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis            }
20055d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis          }
20065d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis        }
20075d04b1af7b76ed536557d4bba24005ad0d2fd608Argyrios Kyrtzidis
20088fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor        if (completionString != NULL) {
20098fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor          printf("\nCompletion string: ");
20108fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor          print_completion_string(completionString, stdout);
20118fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor        }
20128e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        printf("\n");
20138e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        free(Locations[Loc].filename);
20148e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      }
20158e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    }
2016f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
20178e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
2018a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
2019f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  clang_disposeTranslationUnit(TU);
2020f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  clang_disposeIndex(CIdx);
2021f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  free(Locations);
2022f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
2023f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  return 0;
2024f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor}
2025f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor
2026aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic enum CXVisitorResult findFileRefsVisit(void *context,
2027aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                         CXCursor cursor, CXSourceRange range) {
2028aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (clang_Range_isNull(range))
2029aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return CXVisit_Continue;
2030aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2031f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(cursor, NULL);
2032aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintRange(range, "");
2033aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  printf("\n");
2034aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  return CXVisit_Continue;
2035aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis}
2036aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2037aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic int find_file_refs_at(int argc, const char **argv) {
2038aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXIndex CIdx;
2039aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  int errorCode;
2040aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  struct CXUnsavedFile *unsaved_files = 0;
2041aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  int num_unsaved_files = 0;
2042aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXTranslationUnit TU;
2043aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXCursor Cursor;
2044aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CursorSourceLocation *Locations = 0;
2045aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned NumLocations = 0, Loc;
2046aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned Repeats = 1;
2047aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned I;
2048aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2049aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Count the number of locations. */
2050aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  while (strstr(argv[NumLocations+1], "-file-refs-at=") == argv[NumLocations+1])
2051aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    ++NumLocations;
2052aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2053aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Parse the locations. */
2054aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  assert(NumLocations > 0 && "Unable to count locations?");
2055aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  Locations = (CursorSourceLocation *)malloc(
2056aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  NumLocations * sizeof(CursorSourceLocation));
2057aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  for (Loc = 0; Loc < NumLocations; ++Loc) {
2058aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    const char *input = argv[Loc + 1] + strlen("-file-refs-at=");
2059aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if ((errorCode = parse_file_line_column(input, &Locations[Loc].filename,
2060aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                            &Locations[Loc].line,
2061aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                            &Locations[Loc].column, 0, 0)))
2062aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return errorCode;
2063aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2064aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2065aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (parse_remapped_files(argc, argv, NumLocations + 1, &unsaved_files,
2066aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                           &num_unsaved_files))
2067aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return -1;
2068aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2069aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING"))
2070aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    Repeats = 5;
2071aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2072aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Parse the translation unit. When we're testing clang_getCursor() after
2073aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis     reparsing, don't remap unsaved files until the second parse. */
2074aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CIdx = clang_createIndex(1, 1);
2075aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  TU = clang_parseTranslationUnit(CIdx, argv[argc - 1],
2076aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  argv + num_unsaved_files + 1 + NumLocations,
2077aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  argc - num_unsaved_files - 2 - NumLocations,
2078aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  unsaved_files,
2079aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  Repeats > 1? 0 : num_unsaved_files,
2080aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  getDefaultParsingOptions());
2081aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2082aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (!TU) {
2083aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    fprintf(stderr, "unable to parse input\n");
2084aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return -1;
2085aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2086aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2087dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
2088dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return -1;
2089dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2090aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  for (I = 0; I != Repeats; ++I) {
2091aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if (Repeats > 1 &&
2092aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2093aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                     clang_defaultReparseOptions(TU))) {
2094aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      clang_disposeTranslationUnit(TU);
2095aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return 1;
2096aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
2097dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2098dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
2099dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
2100aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2101aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    for (Loc = 0; Loc < NumLocations; ++Loc) {
2102aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      CXFile file = clang_getFile(TU, Locations[Loc].filename);
2103aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (!file)
2104aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        continue;
2105aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2106aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      Cursor = clang_getCursor(TU,
2107aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                               clang_getLocation(TU, file, Locations[Loc].line,
2108aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                                 Locations[Loc].column));
2109dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2110dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      if (checkForErrors(TU) != 0)
2111dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        return -1;
2112dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2113aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (I + 1 == Repeats) {
211426fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen        CXCursorAndRangeVisitor visitor = { 0, findFileRefsVisit };
2115f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko        PrintCursor(Cursor, NULL);
2116aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        printf("\n");
2117aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        clang_findReferencesInFile(Cursor, file, visitor);
2118aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        free(Locations[Loc].filename);
2119dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2120dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        if (checkForErrors(TU) != 0)
2121dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis          return -1;
2122aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      }
2123aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
2124aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
2125aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
2126aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintDiagnostics(TU);
2127aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  clang_disposeTranslationUnit(TU);
2128aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  clang_disposeIndex(CIdx);
2129aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  free(Locations);
2130aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  free_remapped_files(unsaved_files, num_unsaved_files);
2131aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  return 0;
2132aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis}
2133aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
213411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis#define MAX_IMPORTED_ASTFILES 200
213511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
213611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidistypedef struct {
213711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  char **filenames;
213811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  unsigned num_files;
213911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis} ImportedASTFilesData;
214011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
214111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic ImportedASTFilesData *importedASTs_create() {
214211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  ImportedASTFilesData *p;
214311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p = malloc(sizeof(ImportedASTFilesData));
214411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p->filenames = malloc(MAX_IMPORTED_ASTFILES * sizeof(const char *));
214511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p->num_files = 0;
214611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  return p;
214711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis}
214811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
214911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic void importedASTs_dispose(ImportedASTFilesData *p) {
215011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  unsigned i;
215111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (!p)
215211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    return;
215311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
215411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  for (i = 0; i < p->num_files; ++i)
215511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    free(p->filenames[i]);
215611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  free(p->filenames);
215711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  free(p);
215811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis}
215911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
216011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic void importedASTS_insert(ImportedASTFilesData *p, const char *file) {
216111db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  unsigned i;
216211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  assert(p && file);
216311db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  for (i = 0; i < p->num_files; ++i)
216411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    if (strcmp(file, p->filenames[i]) == 0)
216511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis      return;
216611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  assert(p->num_files + 1 < MAX_IMPORTED_ASTFILES);
216711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  p->filenames[p->num_files++] = strdup(file);
216811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis}
216911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
21704e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidistypedef struct {
21714e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *check_prefix;
21724e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  int first_check_printed;
2173dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  int fail_for_error;
21746f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  int abort;
217513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  const char *main_filename;
217611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  ImportedASTFilesData *importedASTs;
21774e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis} IndexData;
21784e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
21794e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void printCheck(IndexData *data) {
21804e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (data->check_prefix) {
21814e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    if (data->first_check_printed) {
21824e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      printf("// %s-NEXT: ", data->check_prefix);
21834e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    } else {
21844e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      printf("// %s     : ", data->check_prefix);
21854e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      data->first_check_printed = 1;
21864e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    }
21874e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
21884e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
21894e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2190dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic void printCXIndexFile(CXIdxClientFile file) {
21914e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString filename = clang_getFileName((CXFile)file);
21924e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("%s", clang_getCString(filename));
21934e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_disposeString(filename);
21944e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
21954e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
219613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidisstatic void printCXIndexLoc(CXIdxLoc loc, CXClientData client_data) {
219713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  IndexData *index_data;
21984e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString filename;
219913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  const char *cname;
2200dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  CXIdxClientFile file;
22014e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  unsigned line, column;
220213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  int isMainFile;
22034e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
220413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
22054e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_indexLoc_getFileLocation(loc, &file, 0, &line, &column, 0);
22064e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (line == 0) {
22078003fd613724847bba834ae41aca6f446af1f818Argyrios Kyrtzidis    printf("<invalid>");
22084e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    return;
22094e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
2210c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  if (!file) {
2211c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis    printf("<no idxfile>");
2212c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis    return;
2213c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  }
22144e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  filename = clang_getFileName((CXFile)file);
22154e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  cname = clang_getCString(filename);
221613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  if (strcmp(cname, index_data->main_filename) == 0)
221713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    isMainFile = 1;
221813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  else
221913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    isMainFile = 0;
222013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  clang_disposeString(filename);
222113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
222213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  if (!isMainFile) {
22234e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printCXIndexFile(file);
22244e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printf(":");
22254e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
22264e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("%d:%d", line, column);
22274e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
22284e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
222913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidisstatic unsigned digitCount(unsigned val) {
223013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  unsigned c = 1;
223113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  while (1) {
223213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    if (val < 10)
223313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis      return c;
223413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    ++c;
223513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    val /= 10;
223613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  }
223713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis}
223813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
22396ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidisstatic CXIdxClientContainer makeClientContainer(const CXIdxEntityInfo *info,
22406ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                                CXIdxLoc loc) {
22414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *name;
22424e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  char *newStr;
2243dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  CXIdxClientFile file;
22444e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  unsigned line, column;
22454e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2246dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  name = info->name;
22474e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (!name)
22484e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    name = "<anon-tag>";
22494e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
22504e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_indexLoc_getFileLocation(loc, &file, 0, &line, &column, 0);
2251f89bc0500c74a590ff9e99aab75bcc22d05bc760Argyrios Kyrtzidis  /* FIXME: free these.*/
225213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  newStr = (char *)malloc(strlen(name) +
225313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis                          digitCount(line) + digitCount(column) + 3);
22544e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  sprintf(newStr, "%s:%d:%d", name, line, column);
22556ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  return (CXIdxClientContainer)newStr;
22564e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
22574e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
22582957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidisstatic void printCXIndexContainer(const CXIdxContainerInfo *info) {
22592957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIdxClientContainer container;
22602957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  container = clang_index_getClientContainer(info);
22613e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis  if (!container)
22623e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis    printf("[<<NULL>>]");
22633e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis  else
22643e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis    printf("[%s]", (const char *)container);
22654e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
22664e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2267dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic const char *getEntityKindString(CXIdxEntityKind kind) {
2268dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  switch (kind) {
2269dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Unexposed: return "<<UNEXPOSED>>";
2270dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Typedef: return "typedef";
2271dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Function: return "function";
2272dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Variable: return "variable";
2273dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Field: return "field";
2274dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_EnumConstant: return "enumerator";
2275dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCClass: return "objc-class";
2276dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCProtocol: return "objc-protocol";
2277dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCCategory: return "objc-category";
2278c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  case CXIdxEntity_ObjCInstanceMethod: return "objc-instance-method";
2279c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  case CXIdxEntity_ObjCClassMethod: return "objc-class-method";
2280dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCProperty: return "objc-property";
2281dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCIvar: return "objc-ivar";
2282dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Enum: return "enum";
2283dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Struct: return "struct";
2284dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Union: return "union";
2285dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_CXXClass: return "c++-class";
22862957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXNamespace: return "namespace";
22872957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXNamespaceAlias: return "namespace-alias";
22882957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXStaticVariable: return "c++-static-var";
22892957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXStaticMethod: return "c++-static-method";
22902957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXInstanceMethod: return "c++-instance-method";
22912957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXConstructor: return "constructor";
22922957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXDestructor: return "destructor";
22932957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXConversionFunction: return "conversion-func";
22942957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXTypeAlias: return "type-alias";
229535adca009c65a9578b91ab0bdca13691624e13afDavid Blaikie  case CXIdxEntity_CXXInterface: return "c++-__interface";
22962957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  }
22972957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  assert(0 && "Garbage entity kind");
22982957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  return 0;
22992957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis}
23002957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis
23012957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidisstatic const char *getEntityTemplateKindString(CXIdxEntityCXXTemplateKind kind) {
23022957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  switch (kind) {
23032957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_NonTemplate: return "";
23042957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_Template: return "-template";
23052957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_TemplatePartialSpecialization:
23062957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    return "-template-partial-spec";
23072957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_TemplateSpecialization: return "-template-spec";
2308dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
23096ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  assert(0 && "Garbage entity kind");
23106ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  return 0;
2311dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis}
2312dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2313838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidisstatic const char *getEntityLanguageString(CXIdxEntityLanguage kind) {
2314838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  switch (kind) {
2315838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_None: return "<none>";
2316838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_C: return "C";
2317838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_ObjC: return "ObjC";
2318838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_CXX: return "C++";
2319838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  }
2320838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  assert(0 && "Garbage language kind");
2321838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  return 0;
2322838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis}
2323838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis
2324dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic void printEntityInfo(const char *cb,
2325dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                            CXClientData client_data,
23266ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                            const CXIdxEntityInfo *info) {
23274e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *name;
23284e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
2329643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  unsigned i;
23304e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
23314e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
23324e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2333c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis  if (!info) {
2334c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis    printf("%s: <<NULL>>", cb);
2335c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis    return;
2336c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis  }
2337c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis
2338dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  name = info->name;
23394e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (!name)
23404e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    name = "<anon-tag>";
23414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
23422957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  printf("%s: kind: %s%s", cb, getEntityKindString(info->kind),
23432957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis         getEntityTemplateKindString(info->templateKind));
2344dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | name: %s", name);
2345dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | USR: %s", info->USR);
2346c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  printf(" | lang: %s", getEntityLanguageString(info->lang));
2347643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis
2348643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  for (i = 0; i != info->numAttributes; ++i) {
2349643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis    const CXIdxAttrInfo *Attr = info->attributes[i];
2350643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis    printf("     <attribute>: ");
2351f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(Attr->cursor, NULL);
2352643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  }
23534e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
23544e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2355b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidisstatic void printBaseClassInfo(CXClientData client_data,
2356b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis                               const CXIdxBaseClassInfo *info) {
2357b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printEntityInfo("     <base>", client_data, info->base);
2358b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printf(" | cursor: ");
2359f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
2360b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printf(" | loc: ");
236113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2362b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis}
2363b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis
2364c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidisstatic void printProtocolList(const CXIdxObjCProtocolRefListInfo *ProtoInfo,
2365c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis                              CXClientData client_data) {
2366c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  unsigned i;
2367c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  for (i = 0; i < ProtoInfo->numProtocols; ++i) {
2368c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printEntityInfo("     <protocol>", client_data,
2369c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis                    ProtoInfo->protocols[i]->protocol);
2370c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf(" | cursor: ");
2371f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(ProtoInfo->protocols[i]->cursor, NULL);
2372c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf(" | loc: ");
237313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    printCXIndexLoc(ProtoInfo->protocols[i]->loc, client_data);
2374c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf("\n");
2375c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  }
2376c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis}
2377c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis
23784e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void index_diagnostic(CXClientData client_data,
2379996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis                             CXDiagnosticSet diagSet, void *reserved) {
23804e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString str;
23814e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *cstr;
2382996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  unsigned numDiags, i;
2383996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  CXDiagnostic diag;
23844e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
23854e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
23864e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
23874e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2388996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  numDiags = clang_getNumDiagnosticsInSet(diagSet);
2389996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  for (i = 0; i != numDiags; ++i) {
2390996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    diag = clang_getDiagnosticInSet(diagSet, i);
2391996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    str = clang_formatDiagnostic(diag, clang_defaultDiagnosticDisplayOptions());
2392996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    cstr = clang_getCString(str);
2393996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    printf("[diagnostic]: %s\n", cstr);
2394996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    clang_disposeString(str);
2395996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis
2396996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    if (getenv("CINDEXTEST_FAILONERROR") &&
2397996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis        clang_getDiagnosticSeverity(diag) >= CXDiagnostic_Error) {
2398996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis      index_data->fail_for_error = 1;
2399996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    }
2400dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
24014e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
24024e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2403dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientFile index_enteredMainFile(CXClientData client_data,
2404dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                                       CXFile file, void *reserved) {
2405dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  IndexData *index_data;
240662d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis  CXString filename;
240762d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis
2408dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
2409dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCheck(index_data);
2410dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
241162d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis  filename = clang_getFileName(file);
241213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  index_data->main_filename = clang_getCString(filename);
241313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  clang_disposeString(filename);
241413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
2415dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf("[enteredMainFile]: ");
2416dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)file);
2417dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf("\n");
2418dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2419dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientFile)file;
24204e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
24214e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2422dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientFile index_ppIncludedFile(CXClientData client_data,
24236ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                            const CXIdxIncludedFileInfo *info) {
24244e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
24254e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
24264e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
24274e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
242866042b32b3b37ddcba731ff05c2792e3bb572102Argyrios Kyrtzidis  printf("[ppIncludedFile]: ");
2429dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)info->file);
24304e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | name: \"%s\"", info->filename);
24314e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | hash loc: ");
243213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->hashLoc, client_data);
24338d7a24e94b58676e57fd3f47353cbdbc59917d81Argyrios Kyrtzidis  printf(" | isImport: %d | isAngled: %d | isModule: %d\n",
24348d7a24e94b58676e57fd3f47353cbdbc59917d81Argyrios Kyrtzidis         info->isImport, info->isAngled, info->isModuleImport);
2435dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2436dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientFile)info->file;
24374e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
24384e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
24392c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidisstatic CXIdxClientFile index_importedASTFile(CXClientData client_data,
24402c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis                                         const CXIdxImportedASTFileInfo *info) {
24412c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  IndexData *index_data;
24422c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
24432c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  printCheck(index_data);
24442c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis
244511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (index_data->importedASTs) {
244611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    CXString filename = clang_getFileName(info->file);
244711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    importedASTS_insert(index_data->importedASTs, clang_getCString(filename));
244811db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    clang_disposeString(filename);
244911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  }
245011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis
24512c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  printf("[importedASTFile]: ");
24522c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)info->file);
2453134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis  if (info->module) {
2454134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    CXString name = clang_Module_getFullName(info->module);
2455134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printf(" | loc: ");
2456134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printCXIndexLoc(info->loc, client_data);
2457134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printf(" | name: \"%s\"", clang_getCString(name));
2458134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    printf(" | isImplicit: %d\n", info->isImplicit);
2459134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis    clang_disposeString(name);
2460900ab95e12bb7483971640a91075699eec391804Argyrios Kyrtzidis  } else {
24613c5527e939c454bfd3e16354908e0c4315be5d2fNAKAMURA Takumi    /* PCH file, the rest are not relevant. */
2462900ab95e12bb7483971640a91075699eec391804Argyrios Kyrtzidis    printf("\n");
2463134d1e8a0b463d929ffeac5eefeae761707bf5d3Argyrios Kyrtzidis  }
24642c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis
24652c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  return (CXIdxClientFile)info->file;
24662c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis}
24672c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis
2468dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientContainer index_startedTranslationUnit(CXClientData client_data,
24694e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis                                                   void *reserved) {
24704e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
24714e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
24724e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
24734e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
247466042b32b3b37ddcba731ff05c2792e3bb572102Argyrios Kyrtzidis  printf("[startedTranslationUnit]\n");
2475dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientContainer)"TU";
24764e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
24774e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
24786ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidisstatic void index_indexDeclaration(CXClientData client_data,
24792957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                                   const CXIdxDeclInfo *info) {
2480dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  IndexData *index_data;
24816ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  const CXIdxObjCCategoryDeclInfo *CatInfo;
24826ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  const CXIdxObjCInterfaceDeclInfo *InterInfo;
2483c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  const CXIdxObjCProtocolRefListInfo *ProtoInfo;
2484792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  const CXIdxObjCPropertyDeclInfo *PropInfo;
2485b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  const CXIdxCXXClassDeclInfo *CXXClassInfo;
2486b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  unsigned i;
2487dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
24884e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2489dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo("[indexDeclaration]", client_data, info->entityInfo);
2490dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | cursor: ");
2491f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
2492dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | loc: ");
249313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2494b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printf(" | semantic-container: ");
2495b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printCXIndexContainer(info->semanticContainer);
2496b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printf(" | lexical-container: ");
2497b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printCXIndexContainer(info->lexicalContainer);
2498dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | isRedecl: %d", info->isRedeclaration);
2499c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isDef: %d", info->isDefinition);
2500838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  if (info->flags & CXIdxDeclFlag_Skipped) {
2501838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    assert(!info->isContainer);
2502838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    printf(" | isContainer: skipped");
2503838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  } else {
2504838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    printf(" | isContainer: %d", info->isContainer);
2505838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  }
2506c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isImplicit: %d\n", info->isImplicit);
25074e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2508b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  for (i = 0; i != info->numAttributes; ++i) {
250987adb0bf9375390de4c66d9e2ad110cc492cd655NAKAMURA Takumi    const CXIdxAttrInfo *Attr = info->attributes[i];
2510b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis    printf("     <attribute>: ");
2511f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(Attr->cursor, NULL);
2512b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis    printf("\n");
2513b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  }
2514b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis
2515dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  if (clang_index_isEntityObjCContainerKind(info->entityInfo->kind)) {
2516dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    const char *kindName = 0;
2517dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    CXIdxObjCContainerKind K = clang_index_getObjCContainerDeclInfo(info)->kind;
2518dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    switch (K) {
2519dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_ForwardRef:
2520dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "forward-ref"; break;
2521dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_Interface:
2522dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "interface"; break;
2523dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_Implementation:
2524dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "implementation"; break;
2525dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    }
2526dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printCheck(index_data);
2527dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printf("     <ObjCContainerInfo>: kind: %s\n", kindName);
2528dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
25294e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
25306ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  if ((CatInfo = clang_index_getObjCCategoryDeclInfo(info))) {
2531dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printEntityInfo("     <ObjCCategoryInfo>: class", client_data,
2532dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                    CatInfo->objcClass);
253321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    printf(" | cursor: ");
2534f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko    PrintCursor(CatInfo->classCursor, NULL);
253521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    printf(" | loc: ");
253613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    printCXIndexLoc(CatInfo->classLoc, client_data);
2537dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printf("\n");
2538dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
25394e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
25406ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  if ((InterInfo = clang_index_getObjCInterfaceDeclInfo(info))) {
25416ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis    if (InterInfo->superInfo) {
2542b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printBaseClassInfo(client_data, InterInfo->superInfo);
25436ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis      printf("\n");
25446ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis    }
25454e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
25464e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2547c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  if ((ProtoInfo = clang_index_getObjCProtocolRefListInfo(info))) {
2548c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printProtocolList(ProtoInfo, client_data);
25496ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  }
25504e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2551792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  if ((PropInfo = clang_index_getObjCPropertyDeclInfo(info))) {
2552792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    if (PropInfo->getter) {
2553792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printEntityInfo("     <getter>", client_data, PropInfo->getter);
2554792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printf("\n");
2555792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    }
2556792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    if (PropInfo->setter) {
2557792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printEntityInfo("     <setter>", client_data, PropInfo->setter);
2558792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printf("\n");
2559792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    }
2560792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  }
2561792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis
2562b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  if ((CXXClassInfo = clang_index_getCXXClassDeclInfo(info))) {
2563b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis    for (i = 0; i != CXXClassInfo->numBases; ++i) {
2564b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printBaseClassInfo(client_data, CXXClassInfo->bases[i]);
2565b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printf("\n");
2566b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis    }
2567b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  }
2568b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis
25692957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  if (info->declAsContainer)
25702957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    clang_index_setClientContainer(info->declAsContainer,
25712957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                              makeClientContainer(info->entityInfo, info->loc));
25724e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
25734e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
25744e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void index_indexEntityReference(CXClientData client_data,
25756ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                       const CXIdxEntityRefInfo *info) {
2576dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo("[indexEntityReference]", client_data, info->referencedEntity);
25774e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | cursor: ");
2578f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  PrintCursor(info->cursor, NULL);
25794e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | loc: ");
258013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2581dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo(" | <parent>:", client_data, info->parentEntity);
25824e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | container: ");
25834e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCXIndexContainer(info->container);
2584c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | refkind: ");
2585aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  switch (info->kind) {
2586aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  case CXIdxEntityRef_Direct: printf("direct"); break;
2587b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  case CXIdxEntityRef_Implicit: printf("implicit"); break;
2588aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  }
25894e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("\n");
25904e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
25914e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
25926f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidisstatic int index_abortQuery(CXClientData client_data, void *reserved) {
25936f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  IndexData *index_data;
25946f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
25956f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  return index_data->abort;
25966f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis}
25976f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis
25984e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic IndexerCallbacks IndexCB = {
25996f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_abortQuery,
26004e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_diagnostic,
2601dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_enteredMainFile,
26024e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_ppIncludedFile,
26032c3e05c266de0d4c465b58ffd129bd0b31604368Argyrios Kyrtzidis  index_importedASTFile,
26044e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_startedTranslationUnit,
2605dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_indexDeclaration,
26064e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_indexEntityReference
26074e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis};
26084e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
260922490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidisstatic unsigned getIndexOptions(void) {
261022490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  unsigned index_opts;
261122490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  index_opts = 0;
261222490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  if (getenv("CINDEXTEST_SUPPRESSREFS"))
261322490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis    index_opts |= CXIndexOpt_SuppressRedundantRefs;
261422490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  if (getenv("CINDEXTEST_INDEXLOCALSYMBOLS"))
261522490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis    index_opts |= CXIndexOpt_IndexFunctionLocalSymbols;
2616838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis  if (!getenv("CINDEXTEST_DISABLE_SKIPPARSEDBODIES"))
2617838eb7e8652e451d93494a4e583e4d11809bcb4aArgyrios Kyrtzidis    index_opts |= CXIndexOpt_SkipParsedBodiesInSession;
261822490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis
261922490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  return index_opts;
262022490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis}
262122490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis
2622d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidisstatic int index_compile_args(int num_args, const char **args,
2623d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              CXIndexAction idxAction,
2624d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              ImportedASTFilesData *importedASTs,
2625d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              const char *check_prefix) {
2626d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  IndexData index_data;
2627d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  unsigned index_opts;
2628d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  int result;
2629d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2630d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (num_args == 0) {
2631d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    fprintf(stderr, "no compiler arguments\n");
2632d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return -1;
2633d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
2634d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2635d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.check_prefix = check_prefix;
2636d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.first_check_printed = 0;
2637d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.fail_for_error = 0;
2638d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.abort = 0;
2639d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.main_filename = "";
2640d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.importedASTs = importedASTs;
2641d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2642d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_opts = getIndexOptions();
2643d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = clang_indexSourceFile(idxAction, &index_data,
2644d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                 &IndexCB,sizeof(IndexCB), index_opts,
2645d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                 0, args, num_args, 0, 0, 0,
2646d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                 getDefaultParsingOptions());
2647d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (index_data.fail_for_error)
2648d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    result = -1;
2649d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2650d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return result;
2651d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis}
2652d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2653d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidisstatic int index_ast_file(const char *ast_file,
2654d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          CXIndex Idx,
2655d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          CXIndexAction idxAction,
2656d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          ImportedASTFilesData *importedASTs,
2657d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          const char *check_prefix) {
2658d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  CXTranslationUnit TU;
2659d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  IndexData index_data;
2660d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  unsigned index_opts;
2661d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  int result;
2662d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2663d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (!CreateTranslationUnit(Idx, ast_file, &TU))
2664d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return -1;
2665d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2666d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.check_prefix = check_prefix;
2667d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.first_check_printed = 0;
2668d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.fail_for_error = 0;
2669d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.abort = 0;
2670d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.main_filename = "";
2671d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_data.importedASTs = importedASTs;
2672d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2673d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  index_opts = getIndexOptions();
2674d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = clang_indexTranslationUnit(idxAction, &index_data,
2675d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                      &IndexCB,sizeof(IndexCB),
2676d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                      index_opts, TU);
2677d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (index_data.fail_for_error)
2678d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    result = -1;
2679d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2680d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  clang_disposeTranslationUnit(TU);
2681d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return result;
2682d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis}
2683d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
268411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisstatic int index_file(int argc, const char **argv, int full) {
26854e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *check_prefix;
26862957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndex Idx;
26872957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndexAction idxAction;
2688d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  ImportedASTFilesData *importedASTs;
2689dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  int result;
26904e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
26914e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  check_prefix = 0;
26924e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc > 0) {
26934e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
26944e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
26954e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      ++argv;
26964e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      --argc;
26974e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    }
26984e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
26994e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
27002957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
27012957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                                /* displayDiagnosics=*/1))) {
27022957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
27032957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    return 1;
27042957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  }
2705d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
2706d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  importedASTs = 0;
270711db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (full)
2708d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    importedASTs = importedASTs_create();
2709d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2710d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = index_compile_args(argc, argv, idxAction, importedASTs, check_prefix);
2711d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (result != 0)
2712d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    goto finished;
2713dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
271411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (full) {
271511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    unsigned i;
2716d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    for (i = 0; i < importedASTs->num_files && result == 0; ++i) {
2717d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      result = index_ast_file(importedASTs->filenames[i], Idx, idxAction,
2718d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                              importedASTs, check_prefix);
271911db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    }
272011db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  }
272121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
272211db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidisfinished:
2723d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  importedASTs_dispose(importedASTs);
27242957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
27252957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_disposeIndex(Idx);
272621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  return result;
272721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis}
272821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
272921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidisstatic int index_tu(int argc, const char **argv) {
2730d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  const char *check_prefix;
273121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  CXIndex Idx;
27322957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndexAction idxAction;
273321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  int result;
273421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
273521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  check_prefix = 0;
273621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc > 0) {
273721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
273821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
273921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      ++argv;
274021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      --argc;
274121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    }
274221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
274321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
2744d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
2745d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                /* displayDiagnosics=*/1))) {
2746d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
2747d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return 1;
2748d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
2749d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
2750d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2751d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  result = index_ast_file(argv[0], Idx, idxAction,
2752d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                          /*importedASTs=*/0, check_prefix);
2753d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2754d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
2755d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  clang_disposeIndex(Idx);
2756d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return result;
2757d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis}
2758d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2759d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidisstatic int index_compile_db(int argc, const char **argv) {
2760d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  const char *check_prefix;
2761d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  CXIndex Idx;
2762d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  CXIndexAction idxAction;
2763d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  int errorCode = 0;
2764d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2765d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  check_prefix = 0;
2766d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (argc > 0) {
2767d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
2768d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
2769d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      ++argv;
2770d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      --argc;
2771d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    }
2772d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
2773d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
277421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc == 0) {
2775d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    fprintf(stderr, "no compilation database\n");
277621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return -1;
277721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
277821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
277921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
278021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis                                /* displayDiagnosics=*/1))) {
278121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
278221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return 1;
278321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
2784d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
278521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
2786d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  {
2787d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    const char *database = argv[0];
2788d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompilationDatabase db = 0;
2789d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompileCommands CCmds = 0;
2790d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompileCommand CCmd;
2791d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXCompilationDatabase_Error ec;
2792d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXString wd;
2793d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis#define MAX_COMPILE_ARGS 512
2794d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    CXString cxargs[MAX_COMPILE_ARGS];
2795d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    const char *args[MAX_COMPILE_ARGS];
2796d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    char *tmp;
2797d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    unsigned len;
2798d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    char *buildDir;
2799d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    int i, a, numCmds, numArgs;
2800d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2801d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    len = strlen(database);
2802d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    tmp = (char *) malloc(len+1);
2803d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    memcpy(tmp, database, len+1);
2804d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    buildDir = dirname(tmp);
2805d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2806d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    db = clang_CompilationDatabase_fromDirectory(buildDir, &ec);
2807d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2808d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    if (db) {
2809d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2810d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      if (ec!=CXCompilationDatabase_NoError) {
2811d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        printf("unexpected error %d code while loading compilation database\n", ec);
2812d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = -1;
2813d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        goto cdb_end;
2814d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
281521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
2816d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      chdir(buildDir);
2817d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      CCmds = clang_CompilationDatabase_getAllCompileCommands(db);
281821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
2819d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      if (!CCmds) {
2820d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        printf("compilation db is empty\n");
2821d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = -1;
2822d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        goto cdb_end;
2823d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
2824d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2825d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      numCmds = clang_CompileCommands_getSize(CCmds);
2826d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2827d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      if (numCmds==0) {
2828d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        fprintf(stderr, "should not get an empty compileCommand set\n");
2829d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = -1;
2830d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        goto cdb_end;
2831d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
2832d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2833d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      for (i=0; i<numCmds && errorCode == 0; ++i) {
2834d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        CCmd = clang_CompileCommands_getCommand(CCmds, i);
2835d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2836d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        wd = clang_CompileCommand_getDirectory(CCmd);
2837d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        chdir(clang_getCString(wd));
2838d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        clang_disposeString(wd);
2839d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2840d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        numArgs = clang_CompileCommand_getNumArgs(CCmd);
2841d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        if (numArgs > MAX_COMPILE_ARGS){
2842d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          fprintf(stderr, "got more compile arguments than maximum\n");
2843d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          errorCode = -1;
2844d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          goto cdb_end;
2845d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        }
2846d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        for (a=0; a<numArgs; ++a) {
2847d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          cxargs[a] = clang_CompileCommand_getArg(CCmd, a);
2848d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          args[a] = clang_getCString(cxargs[a]);
2849d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        }
2850d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2851d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        errorCode = index_compile_args(numArgs, args, idxAction,
2852d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis                                       /*importedASTs=*/0, check_prefix);
2853d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2854d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis        for (a=0; a<numArgs; ++a)
2855d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis          clang_disposeString(cxargs[a]);
2856d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      }
2857d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    } else {
2858d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      printf("database loading failed with error code %d.\n", ec);
2859d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis      errorCode = -1;
2860d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    }
2861d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2862d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  cdb_end:
2863d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    clang_CompileCommands_dispose(CCmds);
2864d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    clang_CompilationDatabase_dispose(db);
2865d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    free(tmp);
2866d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis
2867d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  }
28684e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
28692957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
28702957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_disposeIndex(Idx);
2871d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  return errorCode;
28724e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
28734e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2874fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregorint perform_token_annotation(int argc, const char **argv) {
2875fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  const char *input = argv[1];
2876fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  char *filename = 0;
2877fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned line, second_line;
2878fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned column, second_column;
2879fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXIndex CIdx;
2880fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXTranslationUnit TU = 0;
2881fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  int errorCode;
2882fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
2883fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  int num_unsaved_files = 0;
2884fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXToken *tokens;
2885fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned num_tokens;
2886fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXSourceRange range;
2887fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXSourceLocation startLoc, endLoc;
2888fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXFile file = 0;
2889fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXCursor *cursors = 0;
2890fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned i;
2891fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2892fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  input += strlen("-test-annotate-tokens=");
2893fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if ((errorCode = parse_file_line_column(input, &filename, &line, &column,
2894fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                          &second_line, &second_column)))
2895fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return errorCode;
2896fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2897e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  if (parse_remapped_files(argc, argv, 2, &unsaved_files, &num_unsaved_files)) {
2898e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    free(filename);
2899fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return -1;
2900e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  }
2901fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
29020a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  CIdx = clang_createIndex(0, 1);
2903dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor  TU = clang_parseTranslationUnit(CIdx, argv[argc - 1],
2904dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  argv + num_unsaved_files + 2,
2905dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  argc - num_unsaved_files - 3,
2906dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  unsaved_files,
2907dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  num_unsaved_files,
2908dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  getDefaultParsingOptions());
2909fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!TU) {
2910fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    fprintf(stderr, "unable to parse input\n");
2911fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    clang_disposeIndex(CIdx);
2912fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    free(filename);
2913fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
2914fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return -1;
2915e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
2916fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  errorCode = 0;
2917fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2918e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  if (checkForErrors(TU) != 0) {
2919e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    errorCode = -1;
2920e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    goto teardown;
2921e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  }
2922dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2923ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING")) {
2924ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    for (i = 0; i < 5; ++i) {
2925ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      if (clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2926ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                                       clang_defaultReparseOptions(TU))) {
2927ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        fprintf(stderr, "Unable to reparse translation unit!\n");
2928ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        errorCode = -1;
2929ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        goto teardown;
2930ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      }
2931ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    }
2932ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
2933ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
2934dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
2935dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
2936dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
2937dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
2938dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2939fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  file = clang_getFile(TU, filename);
2940fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!file) {
2941fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    fprintf(stderr, "file %s is not in this translation unit\n", filename);
2942fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
2943fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    goto teardown;
2944fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
2945fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2946fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  startLoc = clang_getLocation(TU, file, line, column);
2947fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (clang_equalLocations(clang_getNullLocation(), startLoc)) {
2948e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(stderr, "invalid source location %s:%d:%d\n", filename, line,
2949fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            column);
2950fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
2951e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    goto teardown;
2952fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
2953fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2954fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  endLoc = clang_getLocation(TU, file, second_line, second_column);
2955fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (clang_equalLocations(clang_getNullLocation(), endLoc)) {
2956e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(stderr, "invalid source location %s:%d:%d\n", filename,
2957fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            second_line, second_column);
2958fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
2959e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    goto teardown;
2960fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
2961fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2962fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  range = clang_getRange(startLoc, endLoc);
2963fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_tokenize(TU, range, &tokens, &num_tokens);
2964dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2965dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
2966dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
2967dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
2968dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
2969dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2970fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  cursors = (CXCursor *)malloc(num_tokens * sizeof(CXCursor));
2971fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_annotateTokens(TU, tokens, num_tokens, cursors);
2972dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2973dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
2974dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
2975dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
2976dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
2977dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2978fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  for (i = 0; i != num_tokens; ++i) {
2979fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *kind = "<unknown>";
2980fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXString spelling = clang_getTokenSpelling(TU, tokens[i]);
2981fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXSourceRange extent = clang_getTokenExtent(TU, tokens[i]);
2982fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    unsigned start_line, start_column, end_line, end_column;
2983fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2984fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    switch (clang_getTokenKind(tokens[i])) {
2985fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Punctuation: kind = "Punctuation"; break;
2986fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Keyword: kind = "Keyword"; break;
2987fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Identifier: kind = "Identifier"; break;
2988fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Literal: kind = "Literal"; break;
2989fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Comment: kind = "Comment"; break;
2990fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
2991a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(clang_getRangeStart(extent),
2992a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              0, &start_line, &start_column, 0);
2993a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(clang_getRangeEnd(extent),
2994a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              0, &end_line, &end_column, 0);
299551b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar    printf("%s: \"%s\" ", kind, clang_getCString(spelling));
2996342742adb69af8ef8f2c652f162496b98c45bc20Benjamin Kramer    clang_disposeString(spelling);
299751b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar    PrintExtent(stdout, start_line, start_column, end_line, end_column);
29980045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    if (!clang_isInvalid(cursors[i].kind)) {
29990045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor      printf(" ");
3000f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko      PrintCursor(cursors[i], NULL);
30010045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    }
30020045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    printf("\n");
3003fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
3004fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free(cursors);
300593f5e6a5d7690f90bc8a94e6b40d6c7d19719e0cTed Kremenek  clang_disposeTokens(TU, tokens, num_tokens);
3006fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3007fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor teardown:
3008a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
3009fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_disposeTranslationUnit(TU);
3010fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_disposeIndex(CIdx);
3011fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free(filename);
3012fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
3013fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  return errorCode;
3014fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
3015fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
3016db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonstatic int
3017db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonperform_test_compilation_db(const char *database, int argc, const char **argv) {
3018db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompilationDatabase db;
3019db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompileCommands CCmds;
3020db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompileCommand CCmd;
3021db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompilationDatabase_Error ec;
3022db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXString wd;
3023db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXString arg;
3024db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  int errorCode = 0;
3025db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  char *tmp;
3026db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  unsigned len;
3027db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  char *buildDir;
3028db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  int i, j, a, numCmds, numArgs;
3029db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3030db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  len = strlen(database);
3031db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  tmp = (char *) malloc(len+1);
3032db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  memcpy(tmp, database, len+1);
3033db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  buildDir = dirname(tmp);
3034db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3035c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison  db = clang_CompilationDatabase_fromDirectory(buildDir, &ec);
3036db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3037db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  if (db) {
3038db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3039db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    if (ec!=CXCompilationDatabase_NoError) {
3040db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      printf("unexpected error %d code while loading compilation database\n", ec);
3041db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      errorCode = -1;
3042db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      goto cdb_end;
3043db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    }
3044db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3045db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    for (i=0; i<argc && errorCode==0; ) {
3046db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      if (strcmp(argv[i],"lookup")==0){
3047c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        CCmds = clang_CompilationDatabase_getCompileCommands(db, argv[i+1]);
3048db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3049db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (!CCmds) {
3050db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("file %s not found in compilation db\n", argv[i+1]);
3051db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          errorCode = -1;
3052db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          break;
3053db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
3054db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3055c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        numCmds = clang_CompileCommands_getSize(CCmds);
3056db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3057db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (numCmds==0) {
3058db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          fprintf(stderr, "should not get an empty compileCommand set for file"
3059db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison                          " '%s'\n", argv[i+1]);
3060db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          errorCode = -1;
3061db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          break;
3062db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
3063db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3064db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        for (j=0; j<numCmds; ++j) {
3065c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          CCmd = clang_CompileCommands_getCommand(CCmds, j);
3066db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3067c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          wd = clang_CompileCommand_getDirectory(CCmd);
3068db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("workdir:'%s'", clang_getCString(wd));
3069db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          clang_disposeString(wd);
3070db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3071db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf(" cmdline:'");
3072c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          numArgs = clang_CompileCommand_getNumArgs(CCmd);
3073db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          for (a=0; a<numArgs; ++a) {
3074db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            if (a) printf(" ");
3075c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison            arg = clang_CompileCommand_getArg(CCmd, a);
3076db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            printf("%s", clang_getCString(arg));
3077db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            clang_disposeString(arg);
3078db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          }
3079db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("'\n");
3080db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
3081db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3082c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        clang_CompileCommands_dispose(CCmds);
3083db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3084db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        i += 2;
3085db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      }
3086db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    }
3087c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison    clang_CompilationDatabase_dispose(db);
3088db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  } else {
3089db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    printf("database loading failed with error code %d.\n", ec);
3090db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    errorCode = -1;
3091db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  }
3092db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3093db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisoncdb_end:
3094db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  free(tmp);
3095db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3096db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  return errorCode;
3097db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison}
3098db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
30990d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
3100f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/* USR printing.                                                              */
3101f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/******************************************************************************/
3102f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3103f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic int insufficient_usr(const char *kind, const char *usage) {
3104f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "USR for '%s' requires: %s\n", kind, usage);
3105f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 1;
3106f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3107f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3108f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic unsigned isUSR(const char *s) {
3109f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return s[0] == 'c' && s[1] == ':';
3110f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3111f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3112f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic int not_usr(const char *s, const char *arg) {
3113f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "'%s' argument ('%s') is not a USR\n", s, arg);
3114f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 1;
3115f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3116f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3117f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic void print_usr(CXString usr) {
3118f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  const char *s = clang_getCString(usr);
3119f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  printf("%s\n", s);
3120f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  clang_disposeString(usr);
3121f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3122f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3123f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic void display_usrs() {
3124f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "-print-usrs options:\n"
3125f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCCategory <class name> <category name>\n"
3126f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCClass <class name>\n"
3127f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCIvar <ivar name> <class USR>\n"
3128f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCMethod <selector> [0=class method|1=instance method] "
3129f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            "<class USR>\n"
3130f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          " ObjCProperty <property name> <class USR>\n"
3131f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          " ObjCProtocol <protocol name>\n");
3132f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3133f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3134f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekint print_usrs(const char **I, const char **E) {
3135f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  while (I != E) {
3136f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    const char *kind = *I;
3137f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    unsigned len = strlen(kind);
3138f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    switch (len) {
3139f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 8:
3140f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCIvar", 8) == 0) {
3141f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
3142f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<ivar name> <class USR>");
3143f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[2]))
3144f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[2]);
3145f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
3146f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
3147a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[2];
3148ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
3149f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCIvar(I[1], x));
3150f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
3151f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3152f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
3153f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3154f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3155f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3156f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 9:
3157f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCClass", 9) == 0) {
3158f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 1 >= E)
3159f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<class name>");
3160f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCClass(I[1]));
3161f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 2;
3162f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3163f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3164f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3165f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 10:
3166f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCMethod", 10) == 0) {
3167f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 3 >= E)
3168f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<method selector> "
3169f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek                "[0=class method|1=instance method] <class USR>");
3170f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[3]))
3171f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[3]);
3172f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
3173f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
3174a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[3];
3175ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
3176f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCMethod(I[1], atoi(I[2]), x));
3177f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
3178f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 4;
3179f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3180f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3181f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3182f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 12:
3183f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCCategory", 12) == 0) {
3184f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
3185f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<class name> <category name>");
3186f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCCategory(I[1], I[2]));
3187f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
3188f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3189f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3190f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCProtocol", 12) == 0) {
3191f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 1 >= E)
3192f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<protocol name>");
3193f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCProtocol(I[1]));
3194f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 2;
3195f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3196f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3197f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCProperty", 12) == 0) {
3198f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
3199f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<property name> <class USR>");
3200f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[2]))
3201f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[2]);
3202f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
3203f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
3204a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[2];
3205ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
3206f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCProperty(I[1], x));
3207f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
3208f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
3209f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
3210f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
3211f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3212f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      default:
3213f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
3214f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
3215f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    break;
3216f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3217f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3218f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  if (I != E) {
3219f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    fprintf(stderr, "Invalid USR kind: %s\n", *I);
3220f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    display_usrs();
3221f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return 1;
3222f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3223f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 0;
3224f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3225f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3226f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekint print_usrs_file(const char *file_name) {
3227f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  char line[2048];
3228f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  const char *args[128];
3229f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  unsigned numChars = 0;
3230f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3231f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  FILE *fp = fopen(file_name, "r");
3232f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  if (!fp) {
3233f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    fprintf(stderr, "error: cannot open '%s'\n", file_name);
3234f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return 1;
3235f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3236f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3237f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  /* This code is not really all that safe, but it works fine for testing. */
3238f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  while (!feof(fp)) {
3239f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    char c = fgetc(fp);
3240f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    if (c == '\n') {
3241f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      unsigned i = 0;
3242f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      const char *s = 0;
3243f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3244f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (numChars == 0)
3245f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        continue;
3246f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3247f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      line[numChars] = '\0';
3248f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      numChars = 0;
3249f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3250f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (line[0] == '/' && line[1] == '/')
3251f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        continue;
3252f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3253f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      s = strtok(line, " ");
3254f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      while (s) {
3255f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        args[i] = s;
3256f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        ++i;
3257f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        s = strtok(0, " ");
3258f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      }
3259f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (print_usrs(&args[0], &args[i]))
3260f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        return 1;
3261f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
3262f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    else
3263f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      line[numChars++] = c;
3264f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3265f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3266f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fclose(fp);
3267f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 0;
3268f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
3269f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
3270f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/******************************************************************************/
32710d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Command line processing.                                                   */
32720d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
32737ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregorint write_pch_file(const char *filename, int argc, const char *argv[]) {
32747ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  CXIndex Idx;
32757ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  CXTranslationUnit TU;
32767ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
32777ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  int num_unsaved_files = 0;
327808aa622ab923fb8a2751f213feee0b9124ee86fbFrancois Pichet  int result = 0;
32797ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
32807ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  Idx = clang_createIndex(/* excludeDeclsFromPCH */1, /* displayDiagnosics=*/1);
32817ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
32827ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
32837ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    clang_disposeIndex(Idx);
32847ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return -1;
32857ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  }
32867ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
32877ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  TU = clang_parseTranslationUnit(Idx, 0,
32887ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  argv + num_unsaved_files,
32897ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  argc - num_unsaved_files,
32907ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  unsaved_files,
32917ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  num_unsaved_files,
3292900ab95e12bb7483971640a91075699eec391804Argyrios Kyrtzidis                                  CXTranslationUnit_Incomplete |
3293900ab95e12bb7483971640a91075699eec391804Argyrios Kyrtzidis                                    CXTranslationUnit_ForSerialization);
32947ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  if (!TU) {
32957ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
32967ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
32977ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    clang_disposeIndex(Idx);
32987ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return 1;
32997ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  }
33007ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
330139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  switch (clang_saveTranslationUnit(TU, filename,
330239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor                                    clang_defaultSaveOptions(TU))) {
330339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_None:
330439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
330539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
330639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_TranslationErrors:
330739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: translation errors\n",
330839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor            filename);
330939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 2;
331039c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
331139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
331239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_InvalidTU:
331339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: invalid translation unit\n",
331439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor            filename);
331539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 3;
331639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
331739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
331839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_Unknown:
331939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  default:
332039c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: unknown error \n", filename);
332139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 1;
332239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
332339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  }
332439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
33257ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  clang_disposeTranslationUnit(TU);
33267ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
33277ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  clang_disposeIndex(Idx);
332839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  return result;
33297ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor}
33307ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
33317ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/******************************************************************************/
3332153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/* Serialized diagnostics.                                                    */
3333153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/******************************************************************************/
3334153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3335153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic const char *getDiagnosticCodeStr(enum CXLoadDiag_Error error) {
3336153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  switch (error) {
3337153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_CannotLoad: return "Cannot Load File";
3338153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_None: break;
3339153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_Unknown: return "Unknown";
3340153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_InvalidFile: return "Invalid File";
3341153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3342153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return "None";
3343153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3344153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3345153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic const char *getSeverityString(enum CXDiagnosticSeverity severity) {
3346153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  switch (severity) {
3347153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Note: return "note";
3348153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Error: return "error";
3349153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Fatal: return "fatal";
3350153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Ignored: return "ignored";
3351153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Warning: return "warning";
3352153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3353153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return "unknown";
3354153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3355153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3356153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printIndent(unsigned indent) {
3357a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  if (indent == 0)
3358a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek    return;
3359a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  fprintf(stderr, "+");
3360a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  --indent;
3361153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  while (indent > 0) {
3362a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek    fprintf(stderr, "-");
3363153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    --indent;
3364153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3365153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3366153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3367153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printLocation(CXSourceLocation L) {
3368153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXFile File;
3369153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXString FileName;
3370153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned line, column, offset;
3371153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3372153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_getExpansionLocation(L, &File, &line, &column, &offset);
3373153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  FileName = clang_getFileName(File);
3374153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3375153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  fprintf(stderr, "%s:%d:%d", clang_getCString(FileName), line, column);
3376153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_disposeString(FileName);
3377153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3378153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3379153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printRanges(CXDiagnostic D, unsigned indent) {
3380153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned i, n = clang_getDiagnosticNumRanges(D);
3381153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3382153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0; i < n; ++i) {
3383153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceLocation Start, End;
3384153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceRange SR = clang_getDiagnosticRange(D, i);
3385153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    Start = clang_getRangeStart(SR);
3386153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    End = clang_getRangeEnd(SR);
3387153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3388153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3389153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "Range: ");
3390153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(Start);
3391153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, " ");
3392153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(End);
3393153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "\n");
3394153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3395153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3396153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3397153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printFixIts(CXDiagnostic D, unsigned indent) {
3398153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned i, n = clang_getDiagnosticNumFixIts(D);
33993739b32a01b9778176757425e62252c288b85383Ted Kremenek  fprintf(stderr, "Number FIXITs = %d\n", n);
3400153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0 ; i < n; ++i) {
3401153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceRange ReplacementRange;
3402153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXString text;
3403153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    text = clang_getDiagnosticFixIt(D, i, &ReplacementRange);
3404153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3405153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3406153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "FIXIT: (");
3407153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(clang_getRangeStart(ReplacementRange));
3408153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, " - ");
3409153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(clang_getRangeEnd(ReplacementRange));
3410153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "): \"%s\"\n", clang_getCString(text));
3411153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(text);
3412153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3413153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3414153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3415153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printDiagnosticSet(CXDiagnosticSet Diags, unsigned indent) {
34169190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi  unsigned i, n;
34179190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi
3418153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (!Diags)
3419153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    return;
3420153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
34219190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi  n = clang_getNumDiagnosticsInSet(Diags);
3422153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0; i < n; ++i) {
3423153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceLocation DiagLoc;
3424153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXDiagnostic D;
3425153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXFile File;
342678d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    CXString FileName, DiagSpelling, DiagOption, DiagCat;
3427153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    unsigned line, column, offset;
342878d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    const char *DiagOptionStr = 0, *DiagCatStr = 0;
3429153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3430153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    D = clang_getDiagnosticInSet(Diags, i);
3431153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagLoc = clang_getDiagnosticLocation(D);
3432153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_getExpansionLocation(DiagLoc, &File, &line, &column, &offset);
3433153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    FileName = clang_getFileName(File);
3434153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagSpelling = clang_getDiagnosticSpelling(D);
3435153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3436153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3437153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3438153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "%s:%d:%d: %s: %s",
3439153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(FileName),
3440153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            line,
3441153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            column,
3442153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            getSeverityString(clang_getDiagnosticSeverity(D)),
3443153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(DiagSpelling));
3444153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3445153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagOption = clang_getDiagnosticOption(D, 0);
3446153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagOptionStr = clang_getCString(DiagOption);
3447153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    if (DiagOptionStr) {
3448153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek      fprintf(stderr, " [%s]", DiagOptionStr);
3449153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    }
3450153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
345178d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    DiagCat = clang_getDiagnosticCategoryText(D);
345278d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    DiagCatStr = clang_getCString(DiagCat);
345378d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    if (DiagCatStr) {
345478d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek      fprintf(stderr, " [%s]", DiagCatStr);
345578d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    }
345678d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek
3457153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "\n");
3458153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3459153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printRanges(D, indent);
3460153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printFixIts(D, indent);
3461153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3462a4ca95a6ebe85d63df08f3290372e97ddaf0b02aNAKAMURA Takumi    /* Print subdiagnostics. */
3463153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printDiagnosticSet(clang_getChildDiagnostics(D), indent+2);
3464153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3465153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(FileName);
3466153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(DiagSpelling);
3467153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(DiagOption);
3468153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3469153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3470153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3471153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic int read_diagnostics(const char *filename) {
3472153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  enum CXLoadDiag_Error error;
3473153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXString errorString;
3474153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXDiagnosticSet Diags = 0;
3475153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3476153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  Diags = clang_loadDiagnostics(filename, &error, &errorString);
3477153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (!Diags) {
3478153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "Trouble deserializing file (%s): %s\n",
3479153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            getDiagnosticCodeStr(error),
3480153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(errorString));
3481153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(errorString);
3482153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    return 1;
3483153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3484153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3485153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  printDiagnosticSet(Diags, 0);
3486a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  fprintf(stderr, "Number of diagnostics: %d\n",
3487a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek          clang_getNumDiagnosticsInSet(Diags));
3488153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_disposeDiagnosticSet(Diags);
3489153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return 0;
3490153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3491153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3492153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/******************************************************************************/
34937ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/* Command line processing.                                                   */
34947ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/******************************************************************************/
3495f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek
3496e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregorstatic CXCursorVisitor GetVisitor(const char *s) {
34977d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  if (s[0] == '\0')
3498e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return FilteredPrintingVisitor;
34997d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  if (strcmp(s, "-usrs") == 0)
35007d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    return USRVisitor;
350159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  if (strncmp(s, "-memory-usage", 13) == 0)
350259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    return GetVisitor(s + 13);
35037d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  return NULL;
35047d40562f83552b7295411e10ee887d8d55470679Ted Kremenek}
35057d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
3506f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenekstatic void print_usage(void) {
3507f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  fprintf(stderr,
35080d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "usage: c-index-test -code-completion-at=<site> <compiler arguments>\n"
35091982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -code-completion-timing=<site> <compiler arguments>\n"
3510f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    "       c-index-test -cursor-at=<site> <compiler arguments>\n"
35113584972866f39d42aa4029586278022d89451bd9NAKAMURA Takumi    "       c-index-test -file-refs-at=<site> <compiler arguments>\n");
35123584972866f39d42aa4029586278022d89451bd9NAKAMURA Takumi  fprintf(stderr,
35134e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    "       c-index-test -index-file [-check-prefix=<FileCheck prefix>] <compiler arguments>\n"
351411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    "       c-index-test -index-file-full [-check-prefix=<FileCheck prefix>] <compiler arguments>\n"
351521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    "       c-index-test -index-tu [-check-prefix=<FileCheck prefix>] <AST file>\n"
3516d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    "       c-index-test -index-compile-db [-check-prefix=<FileCheck prefix>] <compilation database>\n"
35171d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    "       c-index-test -test-file-scan <AST file> <source file> "
351826fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen          "[FileCheck prefix]\n");
351926fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen  fprintf(stderr,
3520fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek    "       c-index-test -test-load-tu <AST file> <symbol filter> "
3521fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek          "[FileCheck prefix]\n"
35227d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    "       c-index-test -test-load-tu-usrs <AST file> <symbol filter> "
35237d40562f83552b7295411e10ee887d8d55470679Ted Kremenek           "[FileCheck prefix]\n"
35241982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -test-load-source <symbol filter> {<args>}*\n");
3525f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  fprintf(stderr,
352659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-load-source-memory-usage "
352759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "<symbol filter> {<args>}*\n"
3528abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    "       c-index-test -test-load-source-reparse <trials> <symbol filter> "
3529abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    "          {<args>}*\n"
35301982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -test-load-source-usrs <symbol filter> {<args>}*\n"
353159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-load-source-usrs-memory-usage "
353259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek          "<symbol filter> {<args>}*\n"
353316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    "       c-index-test -test-annotate-tokens=<range> {<args>}*\n"
353416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    "       c-index-test -test-inclusion-stack-source {<args>}*\n"
35354e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    "       c-index-test -test-inclusion-stack-tu <AST file>\n");
353653513d2e24205965ffd5d4780a10f41cc335f17bChandler Carruth  fprintf(stderr,
35374e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    "       c-index-test -test-print-linkage-source {<args>}*\n"
353859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-print-typekind {<args>}*\n"
35391eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    "       c-index-test -test-print-bitwidth {<args>}*\n"
354059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -print-usr [<CursorKind> {<args>}]*\n"
35417ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    "       c-index-test -print-usr-file <file>\n"
3542153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    "       c-index-test -write-pch <file> <compiler arguments>\n");
3543153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  fprintf(stderr,
3544db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    "       c-index-test -compilation-db [lookup <filename>] database\n");
3545db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  fprintf(stderr,
3546153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    "       c-index-test -read-diagnostics <file>\n\n");
3547caf4bd3d3c66fd95b0159356e2c4da7fbf2897f3Douglas Gregor  fprintf(stderr,
35487d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    " <symbol filter> values:\n%s",
35490d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   all - load all symbols, including those from PCH\n"
35500d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   local - load all symbols except those in PCH\n"
35510d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   category - only load ObjC categories (non-PCH)\n"
35520d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   interface - only load ObjC interfaces (non-PCH)\n"
35530d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   protocol - only load ObjC protocols (non-PCH)\n"
35540d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   function - only load functions (non-PCH)\n"
3555625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    "   typedef - only load typdefs (non-PCH)\n"
3556625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    "   scan-function - scan function bodies (non-PCH)\n\n");
3557f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek}
3558f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek
35596edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/***/
35606edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
35616edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbarint cindextest_main(int argc, const char **argv) {
35620a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  clang_enableStackTraces();
3563153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (argc > 2 && strcmp(argv[1], "-read-diagnostics") == 0)
3564153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek      return read_diagnostics(argv[2]);
3565f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  if (argc > 2 && strstr(argv[1], "-code-completion-at=") == argv[1])
35661982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    return perform_code_completion(argc, argv, 0);
35671982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  if (argc > 2 && strstr(argv[1], "-code-completion-timing=") == argv[1])
35681982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    return perform_code_completion(argc, argv, 1);
3569f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  if (argc > 2 && strstr(argv[1], "-cursor-at=") == argv[1])
3570f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return inspect_cursor_at(argc, argv);
3571aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (argc > 2 && strstr(argv[1], "-file-refs-at=") == argv[1])
3572aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return find_file_refs_at(argc, argv);
35734e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-file") == 0)
357411db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    return index_file(argc - 2, argv + 2, /*full=*/0);
357511db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-file-full") == 0)
357611db1824673fb9834fc414a46fb8da89221c8be3Argyrios Kyrtzidis    return index_file(argc - 2, argv + 2, /*full=*/1);
357721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-tu") == 0)
357821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return index_tu(argc - 2, argv + 2);
3579d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-compile-db") == 0)
3580d10682f7d2944cafb4bc6555df5ab66cd24f8cd8Argyrios Kyrtzidis    return index_compile_db(argc - 2, argv + 2);
35817d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strncmp(argv[1], "-test-load-tu", 13) == 0) {
3582e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 13);
35837d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    if (I)
3584ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek      return perform_test_load_tu(argv[2], argv[3], argc >= 5 ? argv[4] : 0, I,
3585ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                                  NULL);
35867d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  }
3587abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  else if (argc >= 5 && strncmp(argv[1], "-test-load-source-reparse", 25) == 0){
3588abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 25);
3589abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    if (I) {
3590abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      int trials = atoi(argv[2]);
3591abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      return perform_test_reparse_source(argc - 4, argv + 4, trials, argv[3], I,
3592abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                         NULL);
3593abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    }
3594abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
35957d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strncmp(argv[1], "-test-load-source", 17) == 0) {
3596e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 17);
359759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
359859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    PostVisitTU postVisit = 0;
359959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    if (strstr(argv[1], "-memory-usage"))
360059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      postVisit = PrintMemoryUsage;
360159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
36027d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    if (I)
360359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      return perform_test_load_source(argc - 3, argv + 3, argv[2], I,
360459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek                                      postVisit);
36057d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  }
36067d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strcmp(argv[1], "-test-file-scan") == 0)
36071d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    return perform_file_scan(argv[2], argv[3],
36081d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                             argc >= 5 ? argv[4] : 0);
3609fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  else if (argc > 2 && strstr(argv[1], "-test-annotate-tokens=") == argv[1])
3610fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return perform_token_annotation(argc, argv);
361116b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-inclusion-stack-source") == 0)
361216b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all", NULL,
361316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                                    PrintInclusionStack);
361416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-inclusion-stack-tu") == 0)
361516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    return perform_test_load_tu(argv[2], "all", NULL, NULL,
361616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                                PrintInclusionStack);
36173bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-print-linkage-source") == 0)
36183bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all", PrintLinkage,
36193bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek                                    NULL);
36208e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-print-typekind") == 0)
36218e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all",
36228e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek                                    PrintTypeKind, 0);
36231eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko  else if (argc > 2 && strcmp(argv[1], "-test-print-bitwidth") == 0)
36241eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko    return perform_test_load_source(argc - 2, argv + 2, "all",
36251eb60825f0b858a4568c1a9497cc61b0d90c9b3aDmitri Gribenko                                    PrintBitWidth, 0);
3626f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  else if (argc > 1 && strcmp(argv[1], "-print-usr") == 0) {
3627f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    if (argc > 2)
3628f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      return print_usrs(argv + 2, argv + argc);
3629f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    else {
3630f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      display_usrs();
3631f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      return 1;
3632f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
3633f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3634f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-print-usr-file") == 0)
3635f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return print_usrs_file(argv[2]);
36367ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  else if (argc > 2 && strcmp(argv[1], "-write-pch") == 0)
36377ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return write_pch_file(argv[2], argc - 3, argv + 3);
3638db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  else if (argc > 2 && strcmp(argv[1], "-compilation-db") == 0)
3639db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    return perform_test_compilation_db(argv[argc-1], argc - 3, argv + 2);
3640db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3641f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  print_usage();
364250398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff  return 1;
364350398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff}
36446edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
36456edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/***/
36466edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
36476edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/* We intentionally run in a separate thread to ensure we at least minimal
36486edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar * testing of a multithreaded environment (for example, having a reduced stack
36496edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar * size). */
36506edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
36516edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbartypedef struct thread_info {
36526edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  int argc;
36536edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  const char **argv;
36546edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  int result;
36556edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar} thread_info;
365684294911b83003bcac726d00ffa91143752feb57Benjamin Kramervoid thread_runner(void *client_data_v) {
36576edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  thread_info *client_data = client_data_v;
36586edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data->result = cindextest_main(client_data->argc, client_data->argv);
36593be55cdbb5fda326dc55eb01c92d6645a2304309NAKAMURA Takumi#ifdef __CYGWIN__
36603be55cdbb5fda326dc55eb01c92d6645a2304309NAKAMURA Takumi  fflush(stdout);  /* stdout is not flushed on Cygwin. */
36613be55cdbb5fda326dc55eb01c92d6645a2304309NAKAMURA Takumi#endif
36626edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar}
36636edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
36646edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbarint main(int argc, const char **argv) {
3665d1a4f68a4301d1ee3098cc9db0cd507b96dd1beeBenjamin Kramer  thread_info client_data;
3666d1a4f68a4301d1ee3098cc9db0cd507b96dd1beeBenjamin Kramer
3667f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#ifdef CLANG_HAVE_LIBXML
3668f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko  LIBXML_TEST_VERSION
3669f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko#endif
3670f303d4cb10648ac9c2080ae7c9dd507ba615e3a7Dmitri Gribenko
36716160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor  if (getenv("CINDEXTEST_NOTHREADS"))
36726160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor    return cindextest_main(argc, argv);
36736160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor
36746edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data.argc = argc;
36756edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data.argv = argv;
3676a32a6e1abe6972b48b4d5ec79d81b6f7241723bfDaniel Dunbar  clang_executeOnThread(thread_runner, &client_data, 0);
36776edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  return client_data.result;
36786edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar}
3679