c-index-test.c revision 96b098674908eaa59a9128f3305cda6fbbdad563
12b8ee6c2994f738e5162ff46b638974870f51662Steve Naroff/* c-index-test.c */
250398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
350398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff#include "clang-c/Index.h"
4db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison#include "clang-c/CXCompilationDatabase.h"
51e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor#include <ctype.h>
60c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor#include <stdlib.h>
789922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff#include <stdio.h>
8af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff#include <string.h>
9f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor#include <assert.h>
10af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff
110d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
120d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Utility functions.                                                         */
130d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
140d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
152e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#ifdef _MSC_VER
162e06fc877a633abea3b40a64950c7316dac29ca8John Thompsonchar *basename(const char* path)
172e06fc877a633abea3b40a64950c7316dac29ca8John Thompson{
182e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    char* base1 = (char*)strrchr(path, '/');
192e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    char* base2 = (char*)strrchr(path, '\\');
202e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    if (base1 && base2)
212e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return((base1 > base2) ? base1 + 1 : base2 + 1);
222e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    else if (base1)
232e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return(base1 + 1);
242e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    else if (base2)
252e06fc877a633abea3b40a64950c7316dac29ca8John Thompson        return(base2 + 1);
262e06fc877a633abea3b40a64950c7316dac29ca8John Thompson
272e06fc877a633abea3b40a64950c7316dac29ca8John Thompson    return((char*)path);
282e06fc877a633abea3b40a64950c7316dac29ca8John Thompson}
29db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonchar *dirname(char* path)
30db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison{
31db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    char* base1 = (char*)strrchr(path, '/');
32db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    char* base2 = (char*)strrchr(path, '\\');
33db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    if (base1 && base2)
34db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (base1 > base2)
35db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          *base1 = 0;
36db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        else
37db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          *base2 = 0;
38db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    else if (base1)
390fb474a9323525964d437fefa22c06172492d270NAKAMURA Takumi        *base1 = 0;
40db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    else if (base2)
410fb474a9323525964d437fefa22c06172492d270NAKAMURA Takumi        *base2 = 0;
42db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
43db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    return path;
44db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison}
452e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#else
46ff9e18cb3863defa98257500ef49d93a9d2bc216Steve Naroffextern char *basename(const char *);
47db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonextern char *dirname(char *);
482e06fc877a633abea3b40a64950c7316dac29ca8John Thompson#endif
49ff9e18cb3863defa98257500ef49d93a9d2bc216Steve Naroff
5045ba9a1b31110568d0d362c8d31c6133cf9011b7Douglas Gregor/** \brief Return the default parsing options. */
5144c181aec37789f25f6c15543c164416f72e562aDouglas Gregorstatic unsigned getDefaultParsingOptions() {
5244c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  unsigned options = CXTranslationUnit_DetailedPreprocessingRecord;
5344c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
5444c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  if (getenv("CINDEXTEST_EDITING"))
55b1c031be513705d924038f497279b9b599868ba1Douglas Gregor    options |= clang_defaultEditingTranslationUnitOptions();
5687c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor  if (getenv("CINDEXTEST_COMPLETION_CACHING"))
5787c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor    options |= CXTranslationUnit_CacheCompletionResults;
58dcaca015a2f08e03a6d86222fec2e83e04f6b848Argyrios Kyrtzidis  if (getenv("CINDEXTEST_COMPLETION_NO_CACHING"))
59dcaca015a2f08e03a6d86222fec2e83e04f6b848Argyrios Kyrtzidis    options &= ~CXTranslationUnit_CacheCompletionResults;
606a91d385618ea4d28236c496f540a26877c95525Erik Verbruggen  if (getenv("CINDEXTEST_SKIP_FUNCTION_BODIES"))
616a91d385618ea4d28236c496f540a26877c95525Erik Verbruggen    options |= CXTranslationUnit_SkipFunctionBodies;
62d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (getenv("CINDEXTEST_COMPLETION_BRIEF_COMMENTS"))
63d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    options |= CXTranslationUnit_IncludeBriefCommentsInCodeCompletion;
6444c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
6544c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  return options;
6644c181aec37789f25f6c15543c164416f72e562aDouglas Gregor}
6744c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
68bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidisstatic int checkForErrors(CXTranslationUnit TU);
69bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
7051b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbarstatic void PrintExtent(FILE *out, unsigned begin_line, unsigned begin_column,
7151b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar                        unsigned end_line, unsigned end_column) {
7251b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  fprintf(out, "[%d:%d - %d:%d]", begin_line, begin_column,
73d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar          end_line, end_column);
7451b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar}
7551b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar
761c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenekstatic unsigned CreateTranslationUnit(CXIndex Idx, const char *file,
771c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek                                      CXTranslationUnit *TU) {
78e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
79a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  *TU = clang_createTranslationUnit(Idx, file);
806be2a22979e6a4af575af32cccdf6a6190a8edcaDan Gohman  if (!*TU) {
811c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Unable to load translation unit from '%s'!\n", file);
821c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 0;
83e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
841c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  return 1;
851c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
861c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
874db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregorvoid free_remapped_files(struct CXUnsavedFile *unsaved_files,
884db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                         int num_unsaved_files) {
894db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int i;
904db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (i = 0; i != num_unsaved_files; ++i) {
914db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    free((char *)unsaved_files[i].Filename);
924db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    free((char *)unsaved_files[i].Contents);
934db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
94653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor  free(unsaved_files);
954db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor}
964db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
974db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregorint parse_remapped_files(int argc, const char **argv, int start_arg,
984db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                         struct CXUnsavedFile **unsaved_files,
994db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                         int *num_unsaved_files) {
1004db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int i;
1014db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int arg;
1024db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int prefix_len = strlen("-remap-file=");
1034db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *unsaved_files = 0;
1044db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *num_unsaved_files = 0;
105e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1064db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  /* Count the number of remapped files. */
1074db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (arg = start_arg; arg < argc; ++arg) {
1084db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (strncmp(argv[arg], "-remap-file=", prefix_len))
1094db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      break;
110e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1114db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    ++*num_unsaved_files;
1124db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
113e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1144db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  if (*num_unsaved_files == 0)
1154db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    return 0;
116e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1174db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  *unsaved_files
118653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor    = (struct CXUnsavedFile *)malloc(sizeof(struct CXUnsavedFile) *
119653a55f4bc0cb1d0c1f3ff86e9559f744eb3c18fDouglas Gregor                                     *num_unsaved_files);
1204db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (arg = start_arg, i = 0; i != *num_unsaved_files; ++i, ++arg) {
1214db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    struct CXUnsavedFile *unsaved = *unsaved_files + i;
1224db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    const char *arg_string = argv[arg] + prefix_len;
1234db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    int filename_len;
1244db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    char *filename;
1254db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    char *contents;
1264db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    FILE *to_file;
1274db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    const char *semi = strchr(arg_string, ';');
1284db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (!semi) {
129e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr,
1304db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor              "error: -remap-file=from;to argument is missing semicolon\n");
1314db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
1324db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
1334db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
1344db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
1354db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
136e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1374db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Open the file that we're remapping to. */
138c44fe4bf2962bfcb323e93a63ba4a72d67839cdeFrancois Pichet    to_file = fopen(semi + 1, "rb");
1394db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (!to_file) {
1404db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fprintf(stderr, "error: cannot open file %s that we are remapping to\n",
1414db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor              semi + 1);
1424db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
1434db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
1444db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
1454db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
1464db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
147e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1484db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Determine the length of the file we're remapping to. */
1494db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fseek(to_file, 0, SEEK_END);
1504db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Length = ftell(to_file);
1514db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fseek(to_file, 0, SEEK_SET);
152e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1534db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Read the contents of the file we're remapping to. */
1544db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    contents = (char *)malloc(unsaved->Length + 1);
1554db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    if (fread(contents, 1, unsaved->Length, to_file) != unsaved->Length) {
1564db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fprintf(stderr, "error: unexpected %s reading 'to' file %s\n",
1574db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor              (feof(to_file) ? "EOF" : "error"), semi + 1);
1584db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      fclose(to_file);
1594db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      free_remapped_files(*unsaved_files, i);
160e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith      free(contents);
1614db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *unsaved_files = 0;
1624db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      *num_unsaved_files = 0;
1634db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor      return -1;
1644db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    }
1654db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    contents[unsaved->Length] = 0;
1664db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Contents = contents;
167e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1684db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Close the file. */
1694db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    fclose(to_file);
170e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1714db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    /* Copy the file name that we're remapping from. */
1724db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    filename_len = semi - arg_string;
1734db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    filename = (char *)malloc(filename_len + 1);
1744db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    memcpy(filename, arg_string, filename_len);
1754db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    filename[filename_len] = 0;
1764db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    unsaved->Filename = filename;
1774db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
178e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1794db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  return 0;
1804db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor}
1814db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
1820d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
1830d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Pretty-printing.                                                           */
1840d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
1850d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
186ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic const char *FileCheckPrefix = "CHECK";
187ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
188ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCString(const char *CStr) {
1892d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  if (CStr != NULL && CStr[0] != '\0') {
1902d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko    for ( ; *CStr; ++CStr) {
1912d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      const char C = *CStr;
1922d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      switch (C) {
1932d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\n': printf("\\n"); break;
1942d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\r': printf("\\r"); break;
1952d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\t': printf("\\t"); break;
1962d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\v': printf("\\v"); break;
1972d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        case '\f': printf("\\f"); break;
1982d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko        default:   putchar(C);    break;
1992d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko      }
2002d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko    }
2012d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  }
202ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
203ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
204ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCStringWithPrefix(const char *Prefix, const char *CStr) {
205ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf(" %s=[", Prefix);
206ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCString(CStr);
2072d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko  printf("]");
2082d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko}
2092d44d77fed3200e2eff289f55493317e90d3398cDmitri Gribenko
210ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCXStringAndDispose(CXString Str) {
211ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCString(clang_getCString(Str));
212ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  clang_disposeString(Str);
213ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
214ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
215ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCXStringWithPrefixAndDispose(const char *Prefix,
216ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                              CXString Str) {
217ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  PrintCStringWithPrefix(Prefix, clang_getCString(Str));
218ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  clang_disposeString(Str);
219ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
220ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
221430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregorstatic void PrintRange(CXSourceRange R, const char *str) {
222430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  CXFile begin_file, end_file;
223430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  unsigned begin_line, begin_column, end_line, end_column;
224430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
225430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  clang_getSpellingLocation(clang_getRangeStart(R),
226430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                            &begin_file, &begin_line, &begin_column, 0);
227430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  clang_getSpellingLocation(clang_getRangeEnd(R),
228430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                            &end_file, &end_line, &end_column, 0);
229430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  if (!begin_file || !end_file)
230430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    return;
231430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
232ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis  if (str)
233ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis    printf(" %s=", str);
234430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  PrintExtent(stdout, begin_line, begin_column, end_line, end_column);
235430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor}
236430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
237358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregorint want_display_name = 0;
238358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
239cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregorstatic void printVersion(const char *Prefix, CXVersion Version) {
240cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Major < 0)
241cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
242cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf("%s%d", Prefix, Version.Major);
243cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
244cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Minor < 0)
245cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
246cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf(".%d", Version.Minor);
247cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
248cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  if (Version.Subminor < 0)
249cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    return;
250cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor  printf(".%d", Version.Subminor);
251cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor}
252cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
253ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostruct CommentASTDumpingContext {
254ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  int IndentLevel;
255ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko};
256ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
257ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void DumpCXCommentInternal(struct CommentASTDumpingContext *Ctx,
258ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                  CXComment Comment) {
2595ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  unsigned i;
2605ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  unsigned e;
2615ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  enum CXCommentKind Kind = clang_Comment_getKind(Comment);
2625ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko
263ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx->IndentLevel++;
2645ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  for (i = 0, e = Ctx->IndentLevel; i != e; ++i)
265ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("  ");
266ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
267ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("(");
268ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  switch (Kind) {
269ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Null:
270ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Null");
271ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
272ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Text:
273ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Text");
274ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose("Text",
275ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                      clang_TextComment_getText(Comment));
276ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_isWhitespace(Comment))
277ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" IsWhitespace");
278ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
279ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
280ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
281ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_InlineCommand:
282ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_InlineCommand");
283ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
284ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
285ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_InlineCommandComment_getCommandName(Comment));
2862d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    switch (clang_InlineCommandComment_getRenderKind(Comment)) {
2872d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Normal:
2882d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderNormal");
2892d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
2902d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Bold:
2912d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderBold");
2922d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
2932d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Monospaced:
2942d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderMonospaced");
2952d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
2962d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    case CXCommentInlineCommandRenderKind_Emphasized:
2972d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      printf(" RenderEmphasized");
2982d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko      break;
2992d66a5016d4aacce362f89290261c8a1a6eef0d3Dmitri Gribenko    }
3005ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0, e = clang_InlineCommandComment_getNumArgs(Comment);
301ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko         i != e; ++i) {
302ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Arg[%u]=", i);
303ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringAndDispose(
304ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko          clang_InlineCommandComment_getArgText(Comment, i));
305ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
306ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
307ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
308ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
3095ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  case CXComment_HTMLStartTag: {
3105ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    unsigned NumAttrs;
311ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_HTMLStartTag");
312ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
313ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Name",
314ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_HTMLTagComment_getTagName(Comment));
3155ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    NumAttrs = clang_HTMLStartTag_getNumAttrs(Comment);
316ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (NumAttrs != 0) {
317ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Attrs:");
3185ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko      for (i = 0; i != NumAttrs; ++i) {
319ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        printf(" ");
320ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCXStringAndDispose(clang_HTMLStartTag_getAttrName(Comment, i));
321ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        printf("=");
322ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCXStringAndDispose(clang_HTMLStartTag_getAttrValue(Comment, i));
323ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      }
324ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
325ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_HTMLStartTagComment_isSelfClosing(Comment))
326ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" SelfClosing");
327ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
328ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
329ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
3305ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko  }
331ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_HTMLEndTag:
332ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_HTMLEndTag");
333ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
334ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Name",
335ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_HTMLTagComment_getTagName(Comment));
336ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_InlineContentComment_hasTrailingNewline(Comment))
337ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" HasTrailingNewline");
338ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
339ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_Paragraph:
340ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_Paragraph");
341ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_isWhitespace(Comment))
342ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" IsWhitespace");
343ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
344ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_BlockCommand:
345ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_BlockCommand");
346ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
347ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
348ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_BlockCommandComment_getCommandName(Comment));
3495ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0, e = clang_BlockCommandComment_getNumArgs(Comment);
350ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko         i != e; ++i) {
351ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" Arg[%u]=", i);
352ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringAndDispose(
353ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko          clang_BlockCommandComment_getArgText(Comment, i));
354ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
355ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
356ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_ParamCommand:
357ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_ParamCommand");
358ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    switch (clang_ParamCommandComment_getDirection(Comment)) {
359ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_In:
360ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" in");
361ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
362ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_Out:
363ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" out");
364ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
365ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    case CXCommentParamPassDirection_InOut:
366ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" in,out");
367ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      break;
368ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
369ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_ParamCommandComment_isDirectionExplicit(Comment))
370ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" explicitly");
371ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    else
372ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" implicitly");
373ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
374ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "ParamName",
375ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_ParamCommandComment_getParamName(Comment));
376ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_ParamCommandComment_isParamIndexValid(Comment))
377ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" ParamIndex=%u", clang_ParamCommandComment_getParamIndex(Comment));
378ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    else
379ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf(" ParamIndex=Invalid");
380ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
38196b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko  case CXComment_TParamCommand:
38296b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    printf("CXComment_TParamCommand");
38396b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    PrintCXStringWithPrefixAndDispose(
38496b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        "ParamName",
38596b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        clang_TParamCommandComment_getParamName(Comment));
38696b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    if (clang_TParamCommandComment_isParamPositionValid(Comment)) {
38796b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf(" ParamPosition={");
38896b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      for (i = 0, e = clang_TParamCommandComment_getDepth(Comment);
38996b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko           i != e; ++i) {
39096b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        printf("%u", clang_TParamCommandComment_getIndex(Comment, i));
39196b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko        if (i != e - 1)
39296b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko          printf(", ");
39396b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      }
39496b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf("}");
39596b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    } else
39696b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko      printf(" ParamPosition=Invalid");
39796b098674908eaa59a9128f3305cda6fbbdad563Dmitri Gribenko    break;
398ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimBlockCommand:
399ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimBlockCommand");
400ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
401ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "CommandName",
402ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_BlockCommandComment_getCommandName(Comment));
403ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
404ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimBlockLine:
405ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimBlockLine");
406ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
407ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Text",
408ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_VerbatimBlockLineComment_getText(Comment));
409ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
410ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_VerbatimLine:
411ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_VerbatimLine");
412ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCXStringWithPrefixAndDispose(
413ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        "Text",
414ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        clang_VerbatimLineComment_getText(Comment));
415ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
416ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  case CXComment_FullComment:
417ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    printf("CXComment_FullComment");
418ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    break;
419ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
420ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  if (Kind != CXComment_Null) {
421ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const unsigned NumChildren = clang_Comment_getNumChildren(Comment);
4225ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    unsigned i;
4235ef6ea5c69336ea668cd0ff2a2d824f6ce988e1aDmitri Gribenko    for (i = 0; i != NumChildren; ++i) {
424ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      printf("\n// %s: ", FileCheckPrefix);
425ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      DumpCXCommentInternal(Ctx, clang_Comment_getChild(Comment, i));
426ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
427ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
428ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf(")");
429ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx->IndentLevel--;
430ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
431ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
432ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void DumpCXComment(CXComment Comment) {
433ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  struct CommentASTDumpingContext Ctx;
434ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  Ctx.IndentLevel = 1;
435ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("\n// %s:  CommentAST=[\n// %s:", FileCheckPrefix, FileCheckPrefix);
436ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  DumpCXCommentInternal(&Ctx, Comment);
437ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  printf("]");
438ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
439ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
440ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenkostatic void PrintCursorComments(CXCursor Cursor) {
441ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  {
442ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    CXString RawComment;
443ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const char *RawCommentCString;
444ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    CXString BriefComment;
445ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    const char *BriefCommentCString;
446ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
447ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    RawComment = clang_Cursor_getRawCommentText(Cursor);
448ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    RawCommentCString = clang_getCString(RawComment);
449ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (RawCommentCString != NULL && RawCommentCString[0] != '\0') {
450ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCStringWithPrefix("RawComment", RawCommentCString);
451ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintRange(clang_Cursor_getCommentRange(Cursor), "RawCommentRange");
452ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
453ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      BriefComment = clang_Cursor_getBriefCommentText(Cursor);
454ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      BriefCommentCString = clang_getCString(BriefComment);
455ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      if (BriefCommentCString != NULL && BriefCommentCString[0] != '\0')
456ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko        PrintCStringWithPrefix("BriefComment", BriefCommentCString);
457ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      clang_disposeString(BriefComment);
458ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
459ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    clang_disposeString(RawComment);
460ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
461ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
462ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  {
463ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    CXComment Comment = clang_Cursor_getParsedComment(Cursor);
464ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    if (clang_Comment_getKind(Comment) != CXComment_Null) {
465ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      PrintCXStringWithPrefixAndDispose("FullCommentAsHTML",
466ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko                                        clang_FullComment_getAsHTML(Comment));
467ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko      DumpCXComment(Comment);
468ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    }
469ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko  }
470ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko}
471ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko
472aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic void PrintCursor(CXCursor Cursor) {
473aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXTranslationUnit TU = clang_Cursor_getTranslationUnit(Cursor);
474e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (clang_isInvalid(Cursor.kind)) {
475e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    CXString ks = clang_getCursorKindSpelling(Cursor.kind);
476e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("Invalid Cursor => %s", clang_getCString(ks));
477e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    clang_disposeString(ks);
478e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
479699a07d8a0b1579c5178b3baf4bcf9edb6b38108Steve Naroff  else {
480e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    CXString string, ks;
481c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    CXCursor Referenced;
4821db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    unsigned line, column;
483e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    CXCursor SpecializationOf;
4849f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    CXCursor *overridden;
4859f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    unsigned num_overridden;
486430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    unsigned RefNameRangeNr;
487430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CXSourceRange CursorExtent;
488430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CXSourceRange RefNameRange;
489cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int AlwaysUnavailable;
490cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int AlwaysDeprecated;
491cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXString UnavailableMessage;
492cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXString DeprecatedMessage;
493cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    CXPlatformAvailability PlatformAvailability[2];
494cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int NumPlatformAvailability;
495cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    int I;
496aa0cd85838f2a024e589ea4e8c2094130065af21Dmitri Gribenko
497e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    ks = clang_getCursorKindSpelling(Cursor.kind);
498358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    string = want_display_name? clang_getCursorDisplayName(Cursor)
499358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                              : clang_getCursorSpelling(Cursor);
500e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("%s=%s", clang_getCString(ks),
501e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                    clang_getCString(string));
502e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    clang_disposeString(ks);
503ef0cef6cec8de5fc60e469a93436eed7212e0dc2Steve Naroff    clang_disposeString(string);
504e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
505c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    Referenced = clang_getCursorReferenced(Cursor);
506c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    if (!clang_equalCursors(Referenced, clang_getNullCursor())) {
5071f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (clang_getCursorKind(Referenced) == CXCursor_OverloadedDeclRef) {
5081f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        unsigned I, N = clang_getNumOverloadedDecls(Referenced);
5091f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf("[");
5101f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        for (I = 0; I != N; ++I) {
5111f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          CXCursor Ovl = clang_getOverloadedDecl(Referenced, I);
5121f6206ed21e026ca7ddaf0bff9599c476301a695Douglas Gregor          CXSourceLocation Loc;
5131f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          if (I)
5141f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor            printf(", ");
5151f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
5161f6206ed21e026ca7ddaf0bff9599c476301a695Douglas Gregor          Loc = clang_getCursorLocation(Ovl);
517a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor          clang_getSpellingLocation(Loc, 0, &line, &column, 0);
5181f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          printf("%d:%d", line, column);
5191f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        }
5201f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf("]");
5211f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      } else {
5221f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        CXSourceLocation Loc = clang_getCursorLocation(Referenced);
523a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor        clang_getSpellingLocation(Loc, 0, &line, &column, 0);
5241f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        printf(":%d:%d", line, column);
5251f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      }
526c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    }
527b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
528b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (clang_isCursorDefinition(Cursor))
529b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      printf(" (Definition)");
53058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
53158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    switch (clang_getCursorAvailability(Cursor)) {
53258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_Available:
53358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
53458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
53558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_Deprecated:
53658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        printf(" (deprecated)");
53758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
53858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
53958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor      case CXAvailability_NotAvailable:
54058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        printf(" (unavailable)");
54158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        break;
542d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen
543d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen      case CXAvailability_NotAccessible:
544d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen        printf(" (inaccessible)");
545d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen        break;
54658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    }
54795f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
548cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    NumPlatformAvailability
549cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      = clang_getCursorPlatformAvailability(Cursor,
550cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &AlwaysDeprecated,
551cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &DeprecatedMessage,
552cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &AlwaysUnavailable,
553cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            &UnavailableMessage,
554cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                                            PlatformAvailability, 2);
555cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    if (AlwaysUnavailable) {
556cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      printf("  (always unavailable: \"%s\")",
557cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor             clang_getCString(UnavailableMessage));
558cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    } else if (AlwaysDeprecated) {
559cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      printf("  (always deprecated: \"%s\")",
560cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor             clang_getCString(DeprecatedMessage));
561cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    } else {
562cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      for (I = 0; I != NumPlatformAvailability; ++I) {
563cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (I >= 2)
564cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          break;
565cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
566cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        printf("  (%s", clang_getCString(PlatformAvailability[I].Platform));
567cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (PlatformAvailability[I].Unavailable)
568cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printf(", unavailable");
569cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        else {
570cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", introduced=", PlatformAvailability[I].Introduced);
571cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", deprecated=", PlatformAvailability[I].Deprecated);
572cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printVersion(", obsoleted=", PlatformAvailability[I].Obsoleted);
573cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        }
574cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        if (clang_getCString(PlatformAvailability[I].Message)[0])
575cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor          printf(", message=\"%s\"",
576cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor                 clang_getCString(PlatformAvailability[I].Message));
577cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        printf(")");
578cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      }
579cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    }
580cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    for (I = 0; I != NumPlatformAvailability; ++I) {
581cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      if (I >= 2)
582cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor        break;
583cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor      clang_disposeCXPlatformAvailability(PlatformAvailability + I);
584cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    }
585cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
586cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    clang_disposeString(DeprecatedMessage);
587cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor    clang_disposeString(UnavailableMessage);
588cc889664dec7776ebb598e4584e7df5ba2f59ab4Douglas Gregor
589b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor    if (clang_CXXMethod_isStatic(Cursor))
590b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor      printf(" (static)");
591b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor    if (clang_CXXMethod_isVirtual(Cursor))
592b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor      printf(" (virtual)");
593b83d4d7071cea43a3f7b8afc3cda6203c345aa0fDouglas Gregor
59495f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek    if (Cursor.kind == CXCursor_IBOutletCollectionAttr) {
59595f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      CXType T =
59695f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek        clang_getCanonicalType(clang_getIBOutletCollectionType(Cursor));
59795f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      CXString S = clang_getTypeKindSpelling(T.kind);
59895f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      printf(" [IBOutletCollection=%s]", clang_getCString(S));
59995f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek      clang_disposeString(S);
60095f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek    }
6013064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
6023064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    if (Cursor.kind == CXCursor_CXXBaseSpecifier) {
6033064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      enum CX_CXXAccessSpecifier access = clang_getCXXAccessSpecifier(Cursor);
6043064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      unsigned isVirtual = clang_isVirtualBase(Cursor);
6053064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      const char *accessStr = 0;
6063064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
6073064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      switch (access) {
6083064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXInvalidAccessSpecifier:
6093064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "invalid"; break;
6103064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXPublic:
6113064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "public"; break;
6123064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXProtected:
6133064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "protected"; break;
6143064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        case CX_CXXPrivate:
6153064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek          accessStr = "private"; break;
6163064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      }
6173064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
6183064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      printf(" [access=%s isVirtual=%s]", accessStr,
6193064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek             isVirtual ? "true" : "false");
6203064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
621e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor
622e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    SpecializationOf = clang_getSpecializedCursorTemplate(Cursor);
623e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    if (!clang_equalCursors(SpecializationOf, clang_getNullCursor())) {
624e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      CXSourceLocation Loc = clang_getCursorLocation(SpecializationOf);
625e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      CXString Name = clang_getCursorSpelling(SpecializationOf);
626a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor      clang_getSpellingLocation(Loc, 0, &line, &column, 0);
627e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      printf(" [Specialization of %s:%d:%d]",
628e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor             clang_getCString(Name), line, column);
629e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor      clang_disposeString(Name);
630e0329acf5c9437e2086a2fb2bf7a95ae2ac96505Douglas Gregor    }
6319f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
6329f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    clang_getOverriddenCursors(Cursor, &overridden, &num_overridden);
6339f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    if (num_overridden) {
6349f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      unsigned I;
6359f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      printf(" [Overrides ");
6369f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      for (I = 0; I != num_overridden; ++I) {
6379f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor        CXSourceLocation Loc = clang_getCursorLocation(overridden[I]);
638a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor        clang_getSpellingLocation(Loc, 0, &line, &column, 0);
6399f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor        if (I)
6409f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor          printf(", ");
6419f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor        printf("@%d:%d", line, column);
6429f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      }
6439f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      printf("]");
6449f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      clang_disposeOverriddenCursors(overridden);
6459f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    }
646ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
647ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    if (Cursor.kind == CXCursor_InclusionDirective) {
648ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      CXFile File = clang_getIncludedFile(Cursor);
649ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      CXString Included = clang_getFileName(File);
650ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      printf(" (%s)", clang_getCString(Included));
651ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      clang_disposeString(Included);
652dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor
653dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor      if (clang_isFileMultipleIncludeGuarded(TU, File))
654dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor        printf("  [multi-include guarded]");
655ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    }
656430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
657430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    CursorExtent = clang_getCursorExtent(Cursor);
658430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    RefNameRange = clang_getCursorReferenceNameRange(Cursor,
659430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                   CXNameRange_WantQualifier
660430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantSinglePiece
661430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantTemplateArgs,
662430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                     0);
663430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (!clang_equalRanges(CursorExtent, RefNameRange))
664430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      PrintRange(RefNameRange, "SingleRefName");
665430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
666430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    for (RefNameRangeNr = 0; 1; RefNameRangeNr++) {
667430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      RefNameRange = clang_getCursorReferenceNameRange(Cursor,
668430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                   CXNameRange_WantQualifier
669430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                 | CXNameRange_WantTemplateArgs,
670430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                       RefNameRangeNr);
671430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (clang_equalRanges(clang_getNullRange(), RefNameRange))
672430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        break;
673430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (!clang_equalRanges(CursorExtent, RefNameRange))
674430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        PrintRange(RefNameRange, "RefName");
675430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    }
676aa0cd85838f2a024e589ea4e8c2094130065af21Dmitri Gribenko
677ae99b75fbbac1deaccdcc1b326b8fb6b07a1e72dDmitri Gribenko    PrintCursorComments(Cursor);
678699a07d8a0b1579c5178b3baf4bcf9edb6b38108Steve Naroff  }
679af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff}
68089922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff
681e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekstatic const char* GetCursorSource(CXCursor Cursor) {
6821db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  CXSourceLocation Loc = clang_getCursorLocation(Cursor);
68374844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  CXString source;
6841db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  CXFile file;
685b4efaa0a14dd2382aa028c03283b5a7f5345e24dArgyrios Kyrtzidis  clang_getExpansionLocation(Loc, &file, 0, 0, 0);
6861db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  source = clang_getFileName(file);
687e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (!clang_getCString(source)) {
68874844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
68974844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    return "<invalid loc>";
69074844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  }
69174844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  else {
692e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    const char *b = basename(clang_getCString(source));
69374844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
69474844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    return b;
69574844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  }
6969298cfc7475c48fa42b168c37f628663d65ddde7Ted Kremenek}
6979298cfc7475c48fa42b168c37f628663d65ddde7Ted Kremenek
6980d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
699ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/* Callbacks.                                                                 */
700ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/******************************************************************************/
701ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek
702ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenektypedef void (*PostVisitTU)(CXTranslationUnit);
703ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek
704a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregorvoid PrintDiagnostic(CXDiagnostic Diagnostic) {
705a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  FILE *out = stderr;
7065352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  CXFile file;
707274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  CXString Msg;
7080a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  unsigned display_opts = CXDiagnostic_DisplaySourceLocation
709aa5f135f8db82b5e5fb1640fd51f8078e0b2d82dDouglas Gregor    | CXDiagnostic_DisplayColumn | CXDiagnostic_DisplaySourceRanges
710aa5f135f8db82b5e5fb1640fd51f8078e0b2d82dDouglas Gregor    | CXDiagnostic_DisplayOption;
7110a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  unsigned i, num_fixits;
712f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
7130a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (clang_getDiagnosticSeverity(Diagnostic) == CXDiagnostic_Ignored)
7145352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return;
715e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
716274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  Msg = clang_formatDiagnostic(Diagnostic, display_opts);
717274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  fprintf(stderr, "%s\n", clang_getCString(Msg));
718274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor  clang_disposeString(Msg);
719f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
720a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  clang_getSpellingLocation(clang_getDiagnosticLocation(Diagnostic),
721a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                            &file, 0, 0, 0);
7220a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (!file)
7230a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    return;
724e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
7250a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  num_fixits = clang_getDiagnosticNumFixIts(Diagnostic);
7263739b32a01b9778176757425e62252c288b85383Ted Kremenek  fprintf(stderr, "Number FIX-ITs = %d\n", num_fixits);
7270a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  for (i = 0; i != num_fixits; ++i) {
728473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXSourceRange range;
729473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXString insertion_text = clang_getDiagnosticFixIt(Diagnostic, i, &range);
730473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXSourceLocation start = clang_getRangeStart(range);
731473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXSourceLocation end = clang_getRangeEnd(range);
732473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    unsigned start_line, start_column, end_line, end_column;
733473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    CXFile start_file, end_file;
734a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(start, &start_file, &start_line,
735a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              &start_column, 0);
736a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(end, &end_file, &end_line, &end_column, 0);
737473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    if (clang_equalLocations(start, end)) {
738473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Insertion. */
739473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      if (start_file == file)
7400a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Insert \"%s\" at %d:%d\n",
741473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor                clang_getCString(insertion_text), start_line, start_column);
742473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    } else if (strcmp(clang_getCString(insertion_text), "") == 0) {
743473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Removal. */
7440a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      if (start_file == file && end_file == file) {
7450a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Remove ");
7460a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        PrintExtent(out, start_line, start_column, end_line, end_column);
7470a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "\n");
748436f3f0400c633251e4071f81358c47bab964adfDouglas Gregor      }
749473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    } else {
750473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor      /* Replacement. */
7510a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      if (start_file == end_file) {
7520a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        fprintf(out, "FIX-IT: Replace ");
7530a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor        PrintExtent(out, start_line, start_column, end_line, end_column);
754473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor        fprintf(out, " with \"%s\"\n", clang_getCString(insertion_text));
75551c6d384551674facc19f745ecf6e289d28dc55fDouglas Gregor      }
7560a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor      break;
7570a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    }
758473d7019bb54f8a2f0140dca9e9644a935cc6b20Douglas Gregor    clang_disposeString(insertion_text);
75951c6d384551674facc19f745ecf6e289d28dc55fDouglas Gregor  }
7605352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor}
7615352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor
7627473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenekvoid PrintDiagnosticSet(CXDiagnosticSet Set) {
7637473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  int i = 0, n = clang_getNumDiagnosticsInSet(Set);
7647473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  for ( ; i != n ; ++i) {
7657473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    CXDiagnostic Diag = clang_getDiagnosticInSet(Set, i);
7667473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    CXDiagnosticSet ChildDiags = clang_getChildDiagnostics(Diag);
767a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    PrintDiagnostic(Diag);
7687473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek    if (ChildDiags)
7697473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek      PrintDiagnosticSet(ChildDiags);
7707473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  }
7717473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek}
7727473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek
7737473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenekvoid PrintDiagnostics(CXTranslationUnit TU) {
7747473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  CXDiagnosticSet TUSet = clang_getDiagnosticSetFromTU(TU);
7757473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  PrintDiagnosticSet(TUSet);
7767473b1c6e7ba2654d4a0d469198f0e01b485b51aTed Kremenek  clang_disposeDiagnosticSet(TUSet);
777a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor}
778a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor
77959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenekvoid PrintMemoryUsage(CXTranslationUnit TU) {
780b22732365ca26fbccacc4bdd32e875342f571e94Matt Beaumont-Gay  unsigned long total = 0;
7814e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  unsigned i = 0;
782f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  CXTUResourceUsage usage = clang_getCXTUResourceUsage(TU);
7833c683367073e2d98a9046060f9bc7db872a1c63dFrancois Pichet  fprintf(stderr, "Memory usage:\n");
7844e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  for (i = 0 ; i != usage.numEntries; ++i) {
785f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    const char *name = clang_getTUResourceUsageName(usage.entries[i].kind);
78659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    unsigned long amount = usage.entries[i].amount;
78759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    total += amount;
7884e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    fprintf(stderr, "  %s : %ld bytes (%f MBytes)\n", name, amount,
78959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek            ((double) amount)/(1024*1024));
79059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  }
7914e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  fprintf(stderr, "  TOTAL = %ld bytes (%f MBytes)\n", total,
79259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek          ((double) total)/(1024*1024));
793f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  clang_disposeCXTUResourceUsage(usage);
79459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
79559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
796ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek/******************************************************************************/
797e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor/* Logic for testing traversal.                                               */
7980d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
7990d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
800a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregorstatic void PrintCursorExtent(CXCursor C) {
801a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor  CXSourceRange extent = clang_getCursorExtent(C);
802430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  PrintRange(extent, "Extent");
803fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek}
804fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek
805e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor/* Data used by all of the visitors. */
806e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregortypedef struct  {
807e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  CXTranslationUnit TU;
808e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  enum CXCursorKind *Filter;
809e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor} VisitorData;
810fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek
811625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
812e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekenum CXChildVisitResult FilteredPrintingVisitor(CXCursor Cursor,
813e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                CXCursor Parent,
814e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                CXClientData ClientData) {
815e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
816e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  if (!Data->Filter || (Cursor.kind == *(enum CXCursorKind *)Data->Filter)) {
81798258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    CXSourceLocation Loc = clang_getCursorLocation(Cursor);
8181db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    unsigned line, column;
819a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(Loc, 0, &line, &column, 0);
820fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek    printf("// %s: %s:%d:%d: ", FileCheckPrefix,
8211db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor           GetCursorSource(Cursor), line, column);
822aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    PrintCursor(Cursor);
823a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    PrintCursorExtent(Cursor);
824e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("\n");
825e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Recurse;
826625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  }
827e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
828e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
829625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar}
830c857ea4d22e1e6dd9ede1f0e84d48b157c6924fdSteve Naroff
831e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekstatic enum CXChildVisitResult FunctionScanVisitor(CXCursor Cursor,
832e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                   CXCursor Parent,
833e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                                   CXClientData ClientData) {
834625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  const char *startBuf, *endBuf;
835625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  unsigned startLine, startColumn, endLine, endColumn, curLine, curColumn;
836625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXCursor Ref;
837e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
838625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
839b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (Cursor.kind != CXCursor_FunctionDecl ||
840b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      !clang_isCursorDefinition(Cursor))
841e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Continue;
842625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
843625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  clang_getDefinitionSpellingAndExtent(Cursor, &startBuf, &endBuf,
844625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar                                       &startLine, &startColumn,
845625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar                                       &endLine, &endColumn);
846625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  /* Probe the entire body, looking for both decls and refs. */
847625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  curLine = startLine;
848625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  curColumn = startColumn;
849625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
850625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  while (startBuf < endBuf) {
85198258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    CXSourceLocation Loc;
8521db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    CXFile file;
85374844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    CXString source;
854e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
855625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    if (*startBuf == '\n') {
856625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      startBuf++;
857625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curLine++;
858625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curColumn = 1;
859625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    } else if (*startBuf != '\t')
860625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar      curColumn++;
861e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
86298258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    Loc = clang_getCursorLocation(Cursor);
863a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(Loc, &file, 0, 0, 0);
864e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
8651db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor    source = clang_getFileName(file);
866e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    if (clang_getCString(source)) {
867b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor      CXSourceLocation RefLoc
868b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor        = clang_getLocation(Data->TU, file, curLine, curColumn);
869b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor      Ref = clang_getCursor(Data->TU, RefLoc);
87098258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      if (Ref.kind == CXCursor_NoDeclFound) {
87198258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        /* Nothing found here; that's fine. */
87298258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      } else if (Ref.kind != CXCursor_FunctionDecl) {
87398258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        printf("// %s: %s:%d:%d: ", FileCheckPrefix, GetCursorSource(Ref),
87498258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor               curLine, curColumn);
875aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        PrintCursor(Ref);
87698258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor        printf("\n");
87798258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor      }
8784ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff    }
87974844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(source);
880625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    startBuf++;
8812d4d629d8a0de5112c7ae9d05c03ddbf6dcd956aSteve Naroff  }
882e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
883e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
88489922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff}
88550398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
8867d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
8877d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/* USR testing.                                                               */
8887d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
8897d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
890e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregorenum CXChildVisitResult USRVisitor(CXCursor C, CXCursor parent,
891e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor                                   CXClientData ClientData) {
892e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  VisitorData *Data = (VisitorData *)ClientData;
893e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  if (!Data->Filter || (C.kind == *(enum CXCursorKind *)Data->Filter)) {
894cf84aa46def41cccf4dd4c51cd0543b70c11e4ebTed Kremenek    CXString USR = clang_getCursorUSR(C);
895e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    const char *cstr = clang_getCString(USR);
896e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    if (!cstr || cstr[0] == '\0') {
8977d40562f83552b7295411e10ee887d8d55470679Ted Kremenek      clang_disposeString(USR);
898e74ef1289d5fff0a6ea573198bf354fa8cd84d51Ted Kremenek      return CXChildVisit_Recurse;
8997d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    }
900e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek    printf("// %s: %s %s", FileCheckPrefix, GetCursorSource(C), cstr);
901e542f7723a929d42bd9e4dfa526b4ede915b91a7Ted Kremenek
902a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    PrintCursorExtent(C);
9037d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    printf("\n");
9047d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    clang_disposeString(USR);
905e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
906e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return CXChildVisit_Recurse;
907e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
908e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
909e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor  return CXChildVisit_Continue;
9107d40562f83552b7295411e10ee887d8d55470679Ted Kremenek}
9117d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
9127d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
91316b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/* Inclusion stack testing.                                                   */
91416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/******************************************************************************/
91516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
91616b55a71695a33c094383295cc7b7a2080e098daTed Kremenekvoid InclusionVisitor(CXFile includedFile, CXSourceLocation *includeStack,
91716b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                      unsigned includeStackLen, CXClientData data) {
918e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
91916b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  unsigned i;
92074844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  CXString fname;
92174844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek
92274844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  fname = clang_getFileName(includedFile);
923e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  printf("file: %s\nincluded by:\n", clang_getCString(fname));
92474844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  clang_disposeString(fname);
925e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
92616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  for (i = 0; i < includeStackLen; ++i) {
92716b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    CXFile includingFile;
92816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    unsigned line, column;
929a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(includeStack[i], &includingFile, &line,
930a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              &column, 0);
93174844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    fname = clang_getFileName(includingFile);
932e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    printf("  %s:%d:%d\n", clang_getCString(fname), line, column);
93374844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek    clang_disposeString(fname);
93416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  }
93516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  printf("\n");
93616b55a71695a33c094383295cc7b7a2080e098daTed Kremenek}
93716b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
93816b55a71695a33c094383295cc7b7a2080e098daTed Kremenekvoid PrintInclusionStack(CXTranslationUnit TU) {
939e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  clang_getInclusions(TU, InclusionVisitor, NULL);
94016b55a71695a33c094383295cc7b7a2080e098daTed Kremenek}
94116b55a71695a33c094383295cc7b7a2080e098daTed Kremenek
94216b55a71695a33c094383295cc7b7a2080e098daTed Kremenek/******************************************************************************/
9433bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/* Linkage testing.                                                           */
9443bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/******************************************************************************/
9453bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
9463bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenekstatic enum CXChildVisitResult PrintLinkage(CXCursor cursor, CXCursor p,
9473bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek                                            CXClientData d) {
9483bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  const char *linkage = 0;
9493bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
9503bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  if (clang_isInvalid(clang_getCursorKind(cursor)))
9513bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    return CXChildVisit_Recurse;
9523bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
9533bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  switch (clang_getCursorLinkage(cursor)) {
9543bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    case CXLinkage_Invalid: break;
955c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_NoLinkage: linkage = "NoLinkage"; break;
956c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_Internal: linkage = "Internal"; break;
957c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_UniqueExternal: linkage = "UniqueExternal"; break;
958c2a2b3c2c2da1086d76bf804633ed5c6c48ae722Douglas Gregor    case CXLinkage_External: linkage = "External"; break;
9593bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  }
9603bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
9613bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  if (linkage) {
962aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    PrintCursor(cursor);
9633bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    printf("linkage=%s\n", linkage);
9643bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  }
9653bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
9663bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  return CXChildVisit_Recurse;
9673bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek}
9683bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek
9693bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek/******************************************************************************/
9708e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/* Typekind testing.                                                          */
9718e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/******************************************************************************/
9728e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
9738e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenekstatic enum CXChildVisitResult PrintTypeKind(CXCursor cursor, CXCursor p,
9748e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek                                             CXClientData d) {
9758e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  if (!clang_isInvalid(clang_getCursorKind(cursor))) {
9768e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    CXType T = clang_getCursorType(cursor);
9778e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    CXString S = clang_getTypeKindSpelling(T.kind);
978aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    PrintCursor(cursor);
9798e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    printf(" typekind=%s", clang_getCString(S));
980e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isConstQualifiedType(T))
981e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" const");
982e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isVolatileQualifiedType(T))
983e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" volatile");
984e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor    if (clang_isRestrictQualifiedType(T))
985e72fb6f40231a1e8372c7576b69f06f0a1eb28a7Douglas Gregor      printf(" restrict");
9868e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    clang_disposeString(S);
987e1403d2dcb06fc306e186c787a7bd0ca30e06edeBenjamin Kramer    /* Print the canonical type if it is different. */
98804c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    {
98904c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      CXType CT = clang_getCanonicalType(T);
99004c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      if (!clang_equalTypes(T, CT)) {
99104c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        CXString CS = clang_getTypeKindSpelling(CT.kind);
99204c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        printf(" [canonical=%s]", clang_getCString(CS));
99304c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        clang_disposeString(CS);
99404c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      }
99504c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    }
996e1403d2dcb06fc306e186c787a7bd0ca30e06edeBenjamin Kramer    /* Print the return type if it exists. */
99704c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    {
9989a140845438c2fc31e7d48a6dedbc695f4c83c68Ted Kremenek      CXType RT = clang_getCursorResultType(cursor);
99904c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      if (RT.kind != CXType_Invalid) {
100004c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        CXString RS = clang_getTypeKindSpelling(RT.kind);
100104c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        printf(" [result=%s]", clang_getCString(RS));
100204c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek        clang_disposeString(RS);
100304c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek      }
100404c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek    }
1005d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    /* Print the argument types if they exist. */
1006d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    {
1007d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis      int numArgs = clang_Cursor_getNumArguments(cursor);
1008d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis      if (numArgs != -1 && numArgs != 0) {
100947f1165c92bc4104e314223ed9ad251e914687c1Argyrios Kyrtzidis        int i;
1010d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        printf(" [args=");
101147f1165c92bc4104e314223ed9ad251e914687c1Argyrios Kyrtzidis        for (i = 0; i < numArgs; ++i) {
1012d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          CXType T = clang_getCursorType(clang_Cursor_getArgument(cursor, i));
1013d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          if (T.kind != CXType_Invalid) {
1014d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis            CXString S = clang_getTypeKindSpelling(T.kind);
1015d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis            printf(" %s", clang_getCString(S));
1016d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis            clang_disposeString(S);
1017d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis          }
1018d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        }
1019d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis        printf("]");
1020d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis      }
1021d98ef9ae48ab4090d4d5d703ce65cfac62807fdaArgyrios Kyrtzidis    }
10223ce9e7d270e7df86c09c8126b4412d55be7c123bTed Kremenek    /* Print if this is a non-POD type. */
10233ce9e7d270e7df86c09c8126b4412d55be7c123bTed Kremenek    printf(" [isPOD=%d]", clang_isPODType(T));
102404c3cf35a80c09ab78e519f2e71ecccd5c5d8da0Ted Kremenek
10258e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    printf("\n");
10268e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  }
10278e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  return CXChildVisit_Recurse;
10288e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek}
10298e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
10308e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek
10318e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek/******************************************************************************/
10327d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/* Loading ASTs/source.                                                       */
10337d40562f83552b7295411e10ee887d8d55470679Ted Kremenek/******************************************************************************/
10347d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
1035625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbarstatic int perform_test_load(CXIndex Idx, CXTranslationUnit TU,
103698271567b6f10be85bcc75d0bbb67416c8cfb83aTed Kremenek                             const char *filter, const char *prefix,
1037ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             CXCursorVisitor Visitor,
1038ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             PostVisitTU PV) {
1039e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1040fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek  if (prefix)
1041e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    FileCheckPrefix = prefix;
1042e3ee02a324559829142f387f75daf3da118442beTed Kremenek
1043e3ee02a324559829142f387f75daf3da118442beTed Kremenek  if (Visitor) {
1044e3ee02a324559829142f387f75daf3da118442beTed Kremenek    enum CXCursorKind K = CXCursor_NotImplemented;
1045e3ee02a324559829142f387f75daf3da118442beTed Kremenek    enum CXCursorKind *ck = &K;
1046e3ee02a324559829142f387f75daf3da118442beTed Kremenek    VisitorData Data;
1047e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1048e3ee02a324559829142f387f75daf3da118442beTed Kremenek    /* Perform some simple filtering. */
1049e3ee02a324559829142f387f75daf3da118442beTed Kremenek    if (!strcmp(filter, "all") || !strcmp(filter, "local")) ck = NULL;
1050358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    else if (!strcmp(filter, "all-display") ||
1051358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor             !strcmp(filter, "local-display")) {
1052358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      ck = NULL;
1053358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      want_display_name = 1;
1054358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
1055b1ffee6e574d57b79ce2defd1b57052649221c03Daniel Dunbar    else if (!strcmp(filter, "none")) K = (enum CXCursorKind) ~0;
1056e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "category")) K = CXCursor_ObjCCategoryDecl;
1057e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "interface")) K = CXCursor_ObjCInterfaceDecl;
1058e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "protocol")) K = CXCursor_ObjCProtocolDecl;
1059e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "function")) K = CXCursor_FunctionDecl;
1060e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "typedef")) K = CXCursor_TypedefDecl;
1061e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else if (!strcmp(filter, "scan-function")) Visitor = FunctionScanVisitor;
1062e3ee02a324559829142f387f75daf3da118442beTed Kremenek    else {
1063e3ee02a324559829142f387f75daf3da118442beTed Kremenek      fprintf(stderr, "Unknown filter for -test-load-tu: %s\n", filter);
1064e3ee02a324559829142f387f75daf3da118442beTed Kremenek      return 1;
1065e3ee02a324559829142f387f75daf3da118442beTed Kremenek    }
1066e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1067e3ee02a324559829142f387f75daf3da118442beTed Kremenek    Data.TU = TU;
1068e3ee02a324559829142f387f75daf3da118442beTed Kremenek    Data.Filter = ck;
1069e3ee02a324559829142f387f75daf3da118442beTed Kremenek    clang_visitChildren(clang_getTranslationUnitCursor(TU), Visitor, &Data);
1070e3ee02a324559829142f387f75daf3da118442beTed Kremenek  }
1071e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1072ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek  if (PV)
1073ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek    PV(TU);
1074e3ee02a324559829142f387f75daf3da118442beTed Kremenek
1075a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
107616ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
107716ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis    clang_disposeTranslationUnit(TU);
107816ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis    return -1;
107916ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis  }
108016ac8be58801d45358497e90a2f82bc155da8a44Argyrios Kyrtzidis
10810d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek  clang_disposeTranslationUnit(TU);
10820d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek  return 0;
10830d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek}
10840d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
1085fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenekint perform_test_load_tu(const char *file, const char *filter,
1086ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                         const char *prefix, CXCursorVisitor Visitor,
1087ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                         PostVisitTU PV) {
1088625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXIndex Idx;
1089625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar  CXTranslationUnit TU;
1090020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  int result;
1091e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  Idx = clang_createIndex(/* excludeDeclsFromPCH */
10920a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                          !strcmp(filter, "local") ? 1 : 0,
10930a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                          /* displayDiagnosics=*/1);
1094e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1095020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  if (!CreateTranslationUnit(Idx, file, &TU)) {
1096020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
1097625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    return 1;
1098020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  }
1099625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1100020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  result = perform_test_load(Idx, TU, filter, prefix, Visitor, PV);
1101020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  clang_disposeIndex(Idx);
1102020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  return result;
1103625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar}
1104625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar
1105ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenekint perform_test_load_source(int argc, const char **argv,
1106ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             const char *filter, CXCursorVisitor Visitor,
1107ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                             PostVisitTU PV) {
1108ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  CXIndex Idx;
1109ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  CXTranslationUnit TU;
11104db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
11114db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int num_unsaved_files = 0;
11124db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  int result;
1113abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1114ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  Idx = clang_createIndex(/* excludeDeclsFromPCH */
1115358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                          (!strcmp(filter, "local") ||
1116358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                           !strcmp(filter, "local-display"))? 1 : 0,
11174814fb560322daf009fda6226cffccb10bd44620Douglas Gregor                          /* displayDiagnosics=*/0);
1118ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1119020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
1120020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
11214db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor    return -1;
1122020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  }
11234db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor
1124dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor  TU = clang_parseTranslationUnit(Idx, 0,
1125dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  argv + num_unsaved_files,
1126dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  argc - num_unsaved_files,
1127dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  unsaved_files, num_unsaved_files,
1128dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  getDefaultParsingOptions());
1129ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  if (!TU) {
1130ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar    fprintf(stderr, "Unable to load translation unit!\n");
1131abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
1132020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek    clang_disposeIndex(Idx);
1133ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar    return 1;
1134ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar  }
1135ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1136ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek  result = perform_test_load(Idx, TU, filter, NULL, Visitor, PV);
11374db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1138020a095de30ede5b1e3fc65dae501f225a1c89d3Ted Kremenek  clang_disposeIndex(Idx);
11394db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  return result;
1140ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar}
1141ada487d498d82277bbc60312fc6f6479f0afda76Daniel Dunbar
1142abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregorint perform_test_reparse_source(int argc, const char **argv, int trials,
1143abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                const char *filter, CXCursorVisitor Visitor,
1144abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                PostVisitTU PV) {
1145abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  CXIndex Idx;
1146abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  CXTranslationUnit TU;
1147abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
1148abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int num_unsaved_files = 0;
1149abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int result;
1150abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  int trial;
115140098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  int remap_after_trial = 0;
115240098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  char *endptr = 0;
1153abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1154abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  Idx = clang_createIndex(/* excludeDeclsFromPCH */
1155abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                          !strcmp(filter, "local") ? 1 : 0,
11561aa27307c462baaa9e5fda14ff6797dd39fe8b84Douglas Gregor                          /* displayDiagnosics=*/0);
1157abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1158abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
1159abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    clang_disposeIndex(Idx);
1160abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    return -1;
1161abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1162abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1163c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar  /* Load the initial translation unit -- we do this without honoring remapped
1164c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar   * files, so that we have a way to test results after changing the source. */
116544c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  TU = clang_parseTranslationUnit(Idx, 0,
116644c181aec37789f25f6c15543c164416f72e562aDouglas Gregor                                  argv + num_unsaved_files,
116744c181aec37789f25f6c15543c164416f72e562aDouglas Gregor                                  argc - num_unsaved_files,
1168c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar                                  0, 0, getDefaultParsingOptions());
1169abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  if (!TU) {
1170abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
1171abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
1172abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    clang_disposeIndex(Idx);
1173abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    return 1;
1174abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1175abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1176bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
1177bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis    return -1;
1178bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
117940098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  if (getenv("CINDEXTEST_REMAP_AFTER_TRIAL")) {
118040098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis    remap_after_trial =
118140098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis        strtol(getenv("CINDEXTEST_REMAP_AFTER_TRIAL"), &endptr, 10);
118240098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis  }
118340098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis
1184abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  for (trial = 0; trial < trials; ++trial) {
118540098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis    if (clang_reparseTranslationUnit(TU,
118640098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis                             trial >= remap_after_trial ? num_unsaved_files : 0,
118740098e8cd7268535ba581b50af0304b7ecacbef5Argyrios Kyrtzidis                             trial >= remap_after_trial ? unsaved_files : 0,
1188e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregor                                     clang_defaultReparseOptions(TU))) {
1189c8a618002ca57495bf70e055bcafe918859cf4caDaniel Dunbar      fprintf(stderr, "Unable to reparse translation unit!\n");
1190abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      clang_disposeTranslationUnit(TU);
1191abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      free_remapped_files(unsaved_files, num_unsaved_files);
1192abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      clang_disposeIndex(Idx);
1193abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      return -1;
1194abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    }
1195bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
1196bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
1197bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis      return -1;
1198abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
1199abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
1200abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  result = perform_test_load(Idx, TU, filter, NULL, Visitor, PV);
1201bda536df1f5ccd71256eeeab4adbd2cf3769d89eArgyrios Kyrtzidis
1202abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1203abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  clang_disposeIndex(Idx);
1204abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  return result;
1205abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor}
1206abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
12070d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
12081c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/* Logic for testing clang_getCursor().                                       */
12091c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/******************************************************************************/
12101c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
1211dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregorstatic void print_cursor_file_scan(CXTranslationUnit TU, CXCursor cursor,
12121c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek                                   unsigned start_line, unsigned start_col,
12131d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                                   unsigned end_line, unsigned end_col,
12141d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                                   const char *prefix) {
12159096a20fc0fd799911b738ce7a9134dbacaf64b7Ted Kremenek  printf("// %s: ", FileCheckPrefix);
12161d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek  if (prefix)
12171d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    printf("-%s", prefix);
121851b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  PrintExtent(stdout, start_line, start_col, end_line, end_col);
121951b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar  printf(" ");
1220aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintCursor(cursor);
12211c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  printf("\n");
12221c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
12231c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
12241d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenekstatic int perform_file_scan(const char *ast_file, const char *source_file,
12251d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                             const char *prefix) {
12261c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  CXIndex Idx;
12271c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  CXTranslationUnit TU;
12281c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  FILE *fp;
12292389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  CXCursor prevCursor = clang_getNullCursor();
1230b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  CXFile file;
12312389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  unsigned line = 1, col = 1;
12328f0bf81c0f3ab31881cc7db7914e6978bb2a19afDaniel Dunbar  unsigned start_line = 1, start_col = 1;
1233e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
12340a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
12350a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                                /* displayDiagnosics=*/1))) {
12361c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Could not create Index\n");
12371c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
12381c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1239e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
12401c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  if (!CreateTranslationUnit(Idx, ast_file, &TU))
12411c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
1242e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
12431c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  if ((fp = fopen(source_file, "r")) == NULL) {
12441c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    fprintf(stderr, "Could not open '%s'\n", source_file);
12451c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek    return 1;
12461c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1247e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1248b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  file = clang_getFile(TU, source_file);
12492389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar  for (;;) {
12502389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    CXCursor cursor;
12512389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    int c = fgetc(fp);
12522389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar
12532389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if (c == '\n') {
12542389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      ++line;
12552389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      col = 1;
12562389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    } else
12572389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      ++col;
12582389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar
12592389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    /* Check the cursor at this position, and dump the previous one if we have
12602389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar     * found something new.
12612389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar     */
12622389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    cursor = clang_getCursor(TU, clang_getLocation(TU, file, line, col));
12632389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if ((c == EOF || !clang_equalCursors(cursor, prevCursor)) &&
12642389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar        prevCursor.kind != CXCursor_InvalidFile) {
1265dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor      print_cursor_file_scan(TU, prevCursor, start_line, start_col,
1266d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar                             line, col, prefix);
12672389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      start_line = line;
12682389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      start_col = col;
1269a9933b98399f656653a0876fc39e5b9093efb732Benjamin Kramer    }
12702389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    if (c == EOF)
12712389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar      break;
1272a9933b98399f656653a0876fc39e5b9093efb732Benjamin Kramer
12732389eff26431b4f77d3383157adb2c8ccc15ff69Daniel Dunbar    prevCursor = cursor;
12741c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  }
1275e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
12761c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  fclose(fp);
12774f5e21e24fb9e6ec473a13f83b5c9a2c41501a70Douglas Gregor  clang_disposeTranslationUnit(TU);
12784f5e21e24fb9e6ec473a13f83b5c9a2c41501a70Douglas Gregor  clang_disposeIndex(Idx);
12791c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek  return 0;
12801c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek}
12811c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek
12821c6da1735ddc5978a2c24394c5112b4868b347f0Ted Kremenek/******************************************************************************/
128332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor/* Logic for testing clang code completion.                                   */
12840d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
12850d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek
12860c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor/* Parse file:line:column from the input string. Returns 0 on success, non-zero
12870c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor   on failure. If successful, the pointer *filename will contain newly-allocated
12880c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor   memory (that will be owned by the caller) to store the file name. */
1289e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenekint parse_file_line_column(const char *input, char **filename, unsigned *line,
1290fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                           unsigned *column, unsigned *second_line,
1291fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                           unsigned *second_column) {
129288d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor  /* Find the second colon. */
1293fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  const char *last_colon = strrchr(input, ':');
1294fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned values[4], i;
1295fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned num_values = (second_line && second_column)? 4 : 2;
1296fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
12970c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  char *endptr = 0;
1298fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!last_colon || last_colon == input) {
1299fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (num_values == 4)
1300fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      fprintf(stderr, "could not parse filename:line:column:line:column in "
1301fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              "'%s'\n", input);
1302fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    else
1303fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      fprintf(stderr, "could not parse filename:line:column in '%s'\n", input);
13040c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor    return 1;
13050c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
13060c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1307fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  for (i = 0; i != num_values; ++i) {
1308fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *prev_colon;
130988d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor
1310fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    /* Parse the next line or column. */
1311fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    values[num_values - i - 1] = strtol(last_colon + 1, &endptr, 10);
1312fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (*endptr != 0 && *endptr != ':') {
1313e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr, "could not parse %s in '%s'\n",
1314fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              (i % 2 ? "column" : "line"), input);
1315fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      return 1;
1316fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
1317e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1318fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (i + 1 == num_values)
1319fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      break;
132088d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor
1321fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    /* Find the previous colon. */
1322fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    prev_colon = last_colon - 1;
1323fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    while (prev_colon != input && *prev_colon != ':')
1324fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      --prev_colon;
1325fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (prev_colon == input) {
1326e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      fprintf(stderr, "could not parse %s in '%s'\n",
1327fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor              (i % 2 == 0? "column" : "line"), input);
1328e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return 1;
1329fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
1330fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
1331fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    last_colon = prev_colon;
13320c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
1333fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
1334fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *line = values[0];
1335fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *column = values[1];
1336e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1337fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (second_line && second_column) {
1338fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *second_line = values[2];
1339fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *second_column = values[3];
1340fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
1341fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
134288d23952eadb0737e5bd839654d69a0a578e19bcDouglas Gregor  /* Copy the file name. */
1343fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *filename = (char*)malloc(last_colon - input + 1);
1344fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  memcpy(*filename, input, last_colon - input);
1345fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  (*filename)[last_colon - input] = 0;
13460c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  return 0;
13470c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
13480c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
13490c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregorconst char *
13500c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregorclang_getCompletionChunkKindSpelling(enum CXCompletionChunkKind Kind) {
13510c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  switch (Kind) {
13520c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Optional: return "Optional";
13530c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_TypedText: return "TypedText";
13540c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Text: return "Text";
13550c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Placeholder: return "Placeholder";
13560c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Informative: return "Informative";
13570c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_CurrentParameter: return "CurrentParameter";
13580c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftParen: return "LeftParen";
13590c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightParen: return "RightParen";
13600c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftBracket: return "LeftBracket";
13610c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightBracket: return "RightBracket";
13620c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftBrace: return "LeftBrace";
13630c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightBrace: return "RightBrace";
13640c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_LeftAngle: return "LeftAngle";
13650c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_RightAngle: return "RightAngle";
13660c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  case CXCompletionChunk_Comma: return "Comma";
1367ff5ce6eefc7c253ef6edf4d4bfc996fdd82d09aaDouglas Gregor  case CXCompletionChunk_ResultType: return "ResultType";
136801dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_Colon: return "Colon";
136901dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_SemiColon: return "SemiColon";
137001dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_Equal: return "Equal";
137101dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_HorizontalSpace: return "HorizontalSpace";
137201dfea02d1da297e8b53db8eea3d3cc652acda8dDouglas Gregor  case CXCompletionChunk_VerticalSpace: return "VerticalSpace";
13730c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
1374e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
13750c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  return "Unknown";
13760c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
13770c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1378dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidisstatic int checkForErrors(CXTranslationUnit TU) {
1379dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  unsigned Num, i;
1380dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  CXDiagnostic Diag;
1381dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  CXString DiagStr;
1382dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1383dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (!getenv("CINDEXTEST_FAILONERROR"))
1384dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return 0;
1385dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1386dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  Num = clang_getNumDiagnostics(TU);
1387dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  for (i = 0; i != Num; ++i) {
1388dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    Diag = clang_getDiagnostic(TU, i);
1389dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (clang_getDiagnosticSeverity(Diag) >= CXDiagnostic_Error) {
1390dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      DiagStr = clang_formatDiagnostic(Diag,
1391dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis                                       clang_defaultDiagnosticDisplayOptions());
1392dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      fprintf(stderr, "%s\n", clang_getCString(DiagStr));
1393dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      clang_disposeString(DiagStr);
1394dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      clang_disposeDiagnostic(Diag);
1395dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
1396dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    }
1397dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    clang_disposeDiagnostic(Diag);
1398dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
1399dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1400dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  return 0;
1401dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis}
1402dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
14033ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregorvoid print_completion_string(CXCompletionString completion_string, FILE *file) {
1404f8297f1512d29c88f20bf2901f04cc04182a3eeeDaniel Dunbar  int I, N;
1405e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
14063ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  N = clang_getNumCompletionChunks(completion_string);
14070c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  for (I = 0; I != N; ++I) {
14082ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    CXString text;
14092ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    const char *cstr;
14100c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor    enum CXCompletionChunkKind Kind
14113ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      = clang_getCompletionChunkKind(completion_string, I);
1412e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
14133ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor    if (Kind == CXCompletionChunk_Optional) {
14143ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      fprintf(file, "{Optional ");
14153ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      print_completion_string(
1416e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                clang_getCompletionChunkCompletionString(completion_string, I),
14173ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor                              file);
14183ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      fprintf(file, "}");
14193ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor      continue;
14205a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor    }
14215a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor
14225a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor    if (Kind == CXCompletionChunk_VerticalSpace) {
14235a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor      fprintf(file, "{VerticalSpace  }");
14245a9c0bca4504eeda45a3fd0ae1c244b2994f38b2Douglas Gregor      continue;
14253ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor    }
1426e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1427d5a2089663d81faee0de031deabc418fa53ecf3bDouglas Gregor    text = clang_getCompletionChunkText(completion_string, I);
14282ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    cstr = clang_getCString(text);
1429e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(file, "{%s %s}",
14300c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor            clang_getCompletionChunkKindSpelling(Kind),
14312ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek            cstr ? cstr : "");
14322ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek    clang_disposeString(text);
14330c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  }
14342ef6f8f5a35a60870594c5b04e0aa2bf22c6886fTed Kremenek
14353ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor}
14363ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor
14373ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregorvoid print_completion_result(CXCompletionResult *completion_result,
14383ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor                             CXClientData client_data) {
14393ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  FILE *file = (FILE *)client_data;
1440e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  CXString ks = clang_getCursorKindSpelling(completion_result->CursorKind);
14416164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  unsigned annotationCount;
1442ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  enum CXCursorKind ParentKind;
1443ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  CXString ParentName;
1444d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  CXString BriefComment;
1445d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  const char *BriefCommentCString;
1446ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor
1447e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  fprintf(file, "%s:", clang_getCString(ks));
1448e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  clang_disposeString(ks);
1449e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
14503ac738567fdf4df636b9e76c5a2d3dc3d6f80235Douglas Gregor  print_completion_string(completion_result->CompletionString, file);
145158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  fprintf(file, " (%u)",
145212e131385e892e3723483a1081a89bcad29c8a84Douglas Gregor          clang_getCompletionPriority(completion_result->CompletionString));
145358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  switch (clang_getCompletionAvailability(completion_result->CompletionString)){
145458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_Available:
145558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
145658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
145758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_Deprecated:
145858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    fprintf(file, " (deprecated)");
145958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
146058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
146158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  case CXAvailability_NotAvailable:
146258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    fprintf(file, " (unavailable)");
146358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    break;
1464d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen
1465d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen  case CXAvailability_NotAccessible:
1466d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen    fprintf(file, " (inaccessible)");
1467d12059673dcef32bc2b6bae5321654d33863afe6Erik Verbruggen    break;
146858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  }
14696164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen
14706164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  annotationCount = clang_getCompletionNumAnnotations(
14716164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen        completion_result->CompletionString);
14726164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  if (annotationCount) {
14736164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    unsigned i;
14746164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    fprintf(file, " (");
14756164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    for (i = 0; i < annotationCount; ++i) {
14766164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen      if (i != 0)
14776164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen        fprintf(file, ", ");
14786164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen      fprintf(file, "\"%s\"",
14796164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen              clang_getCString(clang_getCompletionAnnotation(
14806164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen                                 completion_result->CompletionString, i)));
14816164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    }
14826164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen    fprintf(file, ")");
14836164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen  }
14846164ea1d75385b6fc3c19e5ab9bb686298436a5aErik Verbruggen
1485ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  if (!getenv("CINDEXTEST_NO_COMPLETION_PARENTS")) {
1486ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    ParentName = clang_getCompletionParent(completion_result->CompletionString,
1487ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor                                           &ParentKind);
1488ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    if (ParentKind != CXCursor_NotImplemented) {
1489ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      CXString KindSpelling = clang_getCursorKindSpelling(ParentKind);
1490ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      fprintf(file, " (parent: %s '%s')",
1491ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor              clang_getCString(KindSpelling),
1492ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor              clang_getCString(ParentName));
1493ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor      clang_disposeString(KindSpelling);
1494ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    }
1495ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor    clang_disposeString(ParentName);
1496ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor  }
1497d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko
1498d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  BriefComment = clang_getCompletionBriefComment(
1499d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko                                        completion_result->CompletionString);
1500d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  BriefCommentCString = clang_getCString(BriefComment);
1501d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (BriefCommentCString && *BriefCommentCString != '\0') {
1502d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    fprintf(file, "(brief comment: %s)", BriefCommentCString);
1503d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  }
1504d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  clang_disposeString(BriefComment);
1505ba1030698dbc276db86b11c5329a1edee8a1805eDouglas Gregor
150658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  fprintf(file, "\n");
15070c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
15080c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
15093da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregorvoid print_completion_contexts(unsigned long long contexts, FILE *file) {
15103da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  fprintf(file, "Completion contexts:\n");
15113da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts == CXCompletionContext_Unknown) {
15123da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Unknown\n");
15133da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15143da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_AnyType) {
15153da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Any type\n");
15163da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15173da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_AnyValue) {
15183da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Any value\n");
15193da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15203da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCObjectValue) {
15213da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C object value\n");
15223da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15233da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCSelectorValue) {
15243da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C selector value\n");
15253da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15263da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_CXXClassTypeValue) {
15273da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "C++ class type value\n");
15283da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15293da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_DotMemberAccess) {
15303da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Dot member access\n");
15313da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15323da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ArrowMemberAccess) {
15333da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Arrow member access\n");
15343da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15353da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCPropertyAccess) {
15363da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C property access\n");
15373da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15383da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_EnumTag) {
15393da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Enum tag\n");
15403da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15413da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_UnionTag) {
15423da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Union tag\n");
15433da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15443da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_StructTag) {
15453da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Struct tag\n");
15463da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15473da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ClassTag) {
15483da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Class name\n");
15493da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15503da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_Namespace) {
15513da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Namespace or namespace alias\n");
15523da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15533da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_NestedNameSpecifier) {
15543da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Nested name specifier\n");
15553da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15563da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCInterface) {
15573da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C interface\n");
15583da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15593da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCProtocol) {
15603da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C protocol\n");
15613da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15623da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCCategory) {
15633da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C category\n");
15643da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15653da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCInstanceMessage) {
15663da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C instance method\n");
15673da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15683da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCClassMessage) {
15693da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C class method\n");
15703da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15713da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_ObjCSelectorName) {
15723da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Objective-C selector name\n");
15733da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15743da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_MacroName) {
15753da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Macro name\n");
15763da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15773da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  if (contexts & CXCompletionContext_NaturalLanguage) {
15783da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    fprintf(file, "Natural language\n");
15793da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor  }
15803da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor}
15813da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
15821e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregorint my_stricmp(const char *s1, const char *s2) {
15831e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  while (*s1 && *s2) {
15846d5552131a4637f9bbe8c93386648e9bbb2c30feNAKAMURA Takumi    int c1 = tolower((unsigned char)*s1), c2 = tolower((unsigned char)*s2);
15851e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    if (c1 < c2)
15861e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      return -1;
15871e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    else if (c1 > c2)
15881e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      return 1;
15891e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
15901e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    ++s1;
15911e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    ++s2;
15921e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  }
15931e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
15941e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  if (*s1)
15951e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    return 1;
15961e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  else if (*s2)
15971e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    return -1;
15981e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor  return 0;
15991e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor}
16001e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
16011982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregorint perform_code_completion(int argc, const char **argv, int timing_only) {
16020c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  const char *input = argv[1];
16030c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  char *filename = 0;
16040c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  unsigned line;
16050c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  unsigned column;
1606f8297f1512d29c88f20bf2901f04cc04182a3eeeDaniel Dunbar  CXIndex CIdx;
1607f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  int errorCode;
1608735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
1609735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  int num_unsaved_files = 0;
1610ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  CXCodeCompleteResults *results = 0;
161125d9b00ab0b128d651d993c38726a00cd9969124Dawn Perchik  CXTranslationUnit TU = 0;
161232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  unsigned I, Repeats = 1;
161332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  unsigned completionOptions = clang_defaultCodeCompleteOptions();
161432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
161532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (getenv("CINDEXTEST_CODE_COMPLETE_PATTERNS"))
161632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    completionOptions |= CXCodeComplete_IncludeCodePatterns;
1617d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko  if (getenv("CINDEXTEST_COMPLETION_BRIEF_COMMENTS"))
1618d99ef536b241071b6f4c01db6525dc03242ac30bDmitri Gribenko    completionOptions |= CXCodeComplete_IncludeBriefComments;
1619df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor
16201982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  if (timing_only)
16211982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    input += strlen("-code-completion-timing=");
16221982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  else
16231982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    input += strlen("-code-completion-at=");
16241982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor
1625e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if ((errorCode = parse_file_line_column(input, &filename, &line, &column,
1626fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                          0, 0)))
1627f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek    return errorCode;
16280c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1629735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  if (parse_remapped_files(argc, argv, 2, &unsaved_files, &num_unsaved_files))
1630735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor    return -1;
1631735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor
163232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  CIdx = clang_createIndex(0, 0);
163332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
163432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (getenv("CINDEXTEST_EDITING"))
163532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    Repeats = 5;
163632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
163732be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  TU = clang_parseTranslationUnit(CIdx, 0,
163832be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                  argv + num_unsaved_files + 2,
163932be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                  argc - num_unsaved_files - 2,
164032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                  0, 0, getDefaultParsingOptions());
164132be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  if (!TU) {
164232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
164332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    return 1;
164432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  }
164508bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor
164608bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor  if (clang_reparseTranslationUnit(TU, 0, 0, clang_defaultReparseOptions(TU))) {
164708bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor    fprintf(stderr, "Unable to reparse translation init!\n");
164808bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor    return 1;
164908bb4c622d0b79c33b4ac78ce1bec79398953daaDouglas Gregor  }
165032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor
165132be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  for (I = 0; I != Repeats; ++I) {
165232be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    results = clang_codeCompleteAt(TU, filename, line, column,
165332be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                   unsaved_files, num_unsaved_files,
165432be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor                                   completionOptions);
165532be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    if (!results) {
165632be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor      fprintf(stderr, "Unable to perform code completion!\n");
16572de41c9c99e0e9ae6488d04c08423a5c1190109eDaniel Dunbar      return 1;
16582de41c9c99e0e9ae6488d04c08423a5c1190109eDaniel Dunbar    }
165932be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor    if (I != Repeats-1)
166032be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor      clang_disposeCodeCompleteResults(results);
166132be4a588fbb87d0d163ead49c42f5438bf0b2b7Douglas Gregor  }
1662936ea3b590117d2cd73b1b92621d06c4a7edbe60Douglas Gregor
1663ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  if (results) {
1664e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    unsigned i, n = results->NumResults, containerIsIncomplete = 0;
16653da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    unsigned long long contexts;
1666e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    enum CXCursorKind containerKind;
16670a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    CXString objCSelector;
16680a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    const char *selectorString;
16691e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    if (!timing_only) {
16701e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      /* Sort the code-completion results based on the typed text. */
16711e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor      clang_sortCodeCompletionResults(results->Results, results->NumResults);
16721e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor
16731982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor      for (i = 0; i != n; ++i)
16741982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor        print_completion_result(results->Results + i, stdout);
16751e5e6684b0f27701e6f7c65f8c6a32a10cbcc3edDouglas Gregor    }
1676a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    n = clang_codeCompleteGetNumDiagnostics(results);
1677a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    for (i = 0; i != n; ++i) {
1678a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      CXDiagnostic diag = clang_codeCompleteGetDiagnostic(results, i);
1679a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      PrintDiagnostic(diag);
1680a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor      clang_disposeDiagnostic(diag);
1681a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor    }
16823da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
16833da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    contexts = clang_codeCompleteGetContexts(results);
16843da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor    print_completion_contexts(contexts, stdout);
16853da626b4f38eb0350de960d71271ca77af7a9cc8Douglas Gregor
16860a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    containerKind = clang_codeCompleteGetContainerKind(results,
16870a47d69af8bda945352997af3da4687a3356096aDouglas Gregor                                                       &containerIsIncomplete);
1688e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
1689e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    if (containerKind != CXCursor_InvalidCode) {
1690e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      /* We have found a container */
1691e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      CXString containerUSR, containerKindSpelling;
1692e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      containerKindSpelling = clang_getCursorKindSpelling(containerKind);
1693e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      printf("Container Kind: %s\n", clang_getCString(containerKindSpelling));
1694e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      clang_disposeString(containerKindSpelling);
1695e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
1696e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      if (containerIsIncomplete) {
1697e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor        printf("Container is incomplete\n");
1698e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      }
1699e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      else {
1700e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor        printf("Container is complete\n");
1701e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      }
1702e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
1703e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      containerUSR = clang_codeCompleteGetContainerUSR(results);
1704e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      printf("Container USR: %s\n", clang_getCString(containerUSR));
1705e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor      clang_disposeString(containerUSR);
1706e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor    }
1707e081a61bb0dc546fd623bf259435d17c9a4ea0d5Douglas Gregor
17080a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    objCSelector = clang_codeCompleteGetObjCSelector(results);
17090a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    selectorString = clang_getCString(objCSelector);
17100a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    if (selectorString && strlen(selectorString) > 0) {
17110a47d69af8bda945352997af3da4687a3356096aDouglas Gregor      printf("Objective-C selector: %s\n", selectorString);
17120a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    }
17130a47d69af8bda945352997af3da4687a3356096aDouglas Gregor    clang_disposeString(objCSelector);
17140a47d69af8bda945352997af3da4687a3356096aDouglas Gregor
1715ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor    clang_disposeCodeCompleteResults(results);
1716ec6762c709726bf2ee5f76c21df81e71a56e6f81Douglas Gregor  }
1717df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor  clang_disposeTranslationUnit(TU);
17180c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  clang_disposeIndex(CIdx);
17190c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor  free(filename);
1720e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1721735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1722735df88a38e80c1ca70daa889aa516b8b9f54b50Douglas Gregor
1723f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  return 0;
17240c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor}
17250c8296dfb495f41d6f0de6fe1d03014ffd063674Douglas Gregor
1726f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregortypedef struct {
1727f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  char *filename;
1728f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned line;
1729f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned column;
1730f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor} CursorSourceLocation;
1731f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor
1732aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic int inspect_cursor_at(int argc, const char **argv) {
1733f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXIndex CIdx;
1734f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  int errorCode;
1735f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
1736f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  int num_unsaved_files = 0;
1737f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXTranslationUnit TU;
1738f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CXCursor Cursor;
1739f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  CursorSourceLocation *Locations = 0;
1740f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  unsigned NumLocations = 0, Loc;
17418e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  unsigned Repeats = 1;
1742bdc4b366e80c125184a3b3c56fa4619cb4ac9e45Douglas Gregor  unsigned I;
17438e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
1744e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  /* Count the number of locations. */
1745f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  while (strstr(argv[NumLocations+1], "-cursor-at=") == argv[NumLocations+1])
1746f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    ++NumLocations;
1747e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1748f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  /* Parse the locations. */
1749f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  assert(NumLocations > 0 && "Unable to count locations?");
1750f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  Locations = (CursorSourceLocation *)malloc(
1751f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                                  NumLocations * sizeof(CursorSourceLocation));
1752f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  for (Loc = 0; Loc < NumLocations; ++Loc) {
1753f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    const char *input = argv[Loc + 1] + strlen("-cursor-at=");
1754e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    if ((errorCode = parse_file_line_column(input, &Locations[Loc].filename,
1755e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek                                            &Locations[Loc].line,
1756fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                            &Locations[Loc].column, 0, 0)))
1757f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor      return errorCode;
1758f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
1759e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1760e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  if (parse_remapped_files(argc, argv, NumLocations + 1, &unsaved_files,
1761f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                           &num_unsaved_files))
1762f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return -1;
1763e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
17648e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  if (getenv("CINDEXTEST_EDITING"))
17658e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    Repeats = 5;
17668e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
17678e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  /* Parse the translation unit. When we're testing clang_getCursor() after
17688e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor     reparsing, don't remap unsaved files until the second parse. */
17698e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  CIdx = clang_createIndex(1, 1);
17708e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor  TU = clang_parseTranslationUnit(CIdx, argv[argc - 1],
17718e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                  argv + num_unsaved_files + 1 + NumLocations,
1772f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor                                  argc - num_unsaved_files - 2 - NumLocations,
17738e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                  unsaved_files,
17748e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                  Repeats > 1? 0 : num_unsaved_files,
17758e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                  getDefaultParsingOptions());
17768e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
1777f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  if (!TU) {
1778f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    fprintf(stderr, "unable to parse input\n");
1779f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return -1;
1780f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
1781e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
1782dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
1783dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return -1;
1784dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1785bdc4b366e80c125184a3b3c56fa4619cb4ac9e45Douglas Gregor  for (I = 0; I != Repeats; ++I) {
17868e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    if (Repeats > 1 &&
17878e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
17888e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                     clang_defaultReparseOptions(TU))) {
17898e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      clang_disposeTranslationUnit(TU);
17908e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      return 1;
17918e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    }
1792dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1793dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
1794dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
17958e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
17968e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    for (Loc = 0; Loc < NumLocations; ++Loc) {
17978e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      CXFile file = clang_getFile(TU, Locations[Loc].filename);
17988e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      if (!file)
17998e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        continue;
18008e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
18018e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      Cursor = clang_getCursor(TU,
18028e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                               clang_getLocation(TU, file, Locations[Loc].line,
18038e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor                                                 Locations[Loc].column));
1804dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1805dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      if (checkForErrors(TU) != 0)
1806dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        return -1;
1807dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
18088e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      if (I + 1 == Repeats) {
18098fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor        CXCompletionString completionString = clang_getCursorCompletionString(
18108fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor                                                                        Cursor);
181166373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        CXSourceLocation CursorLoc = clang_getCursorLocation(Cursor);
181266373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        CXString Spelling;
181366373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        const char *cspell;
181466373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        unsigned line, column;
181566373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        clang_getSpellingLocation(CursorLoc, 0, &line, &column, 0);
181666373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        printf("%d:%d ", line, column);
1817aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        PrintCursor(Cursor);
181866373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        PrintCursorExtent(Cursor);
181966373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        Spelling = clang_getCursorSpelling(Cursor);
182066373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        cspell = clang_getCString(Spelling);
1821ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis        if (cspell && strlen(cspell) != 0) {
1822ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          unsigned pieceIndex;
1823ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          printf(" Spelling=%s (", cspell);
1824ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          for (pieceIndex = 0; ; ++pieceIndex) {
18256c235bc9e126fcecd16257838aa040fc9c363328Benjamin Kramer            CXSourceRange range =
18266c235bc9e126fcecd16257838aa040fc9c363328Benjamin Kramer              clang_Cursor_getSpellingNameRange(Cursor, pieceIndex, 0);
1827ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis            if (clang_Range_isNull(range))
1828ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis              break;
1829ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis            PrintRange(range, 0);
1830ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          }
1831ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis          printf(")");
1832ba1da14e79c1ecd49306e5175b69baa1e7ed4293Argyrios Kyrtzidis        }
183366373dd2d508407ed5894ad65f654eea8d892319Argyrios Kyrtzidis        clang_disposeString(Spelling);
183434ebe1e1b0779bcea2f277bc6b4e9dd98bf70b7bArgyrios Kyrtzidis        if (clang_Cursor_getObjCSelectorIndex(Cursor) != -1)
183534ebe1e1b0779bcea2f277bc6b4e9dd98bf70b7bArgyrios Kyrtzidis          printf(" Selector index=%d",clang_Cursor_getObjCSelectorIndex(Cursor));
1836f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis        if (clang_Cursor_isDynamicCall(Cursor))
1837f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis          printf(" Dynamic-call");
1838f39a7aea7dd0bf0716a066e2db2f97ea8730e4faArgyrios Kyrtzidis
18398fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor        if (completionString != NULL) {
18408fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor          printf("\nCompletion string: ");
18418fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor          print_completion_string(completionString, stdout);
18428fa0a80b4482ad94e82c4a19e23de17fd69140b5Douglas Gregor        }
18438e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        printf("\n");
18448e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor        free(Locations[Loc].filename);
18458e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor      }
18468e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor    }
1847f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  }
18488e08dec6374473b11c019035aef130e0bb0551f9Douglas Gregor
1849a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
1850f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  clang_disposeTranslationUnit(TU);
1851f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  clang_disposeIndex(CIdx);
1852f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  free(Locations);
1853f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
1854f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  return 0;
1855f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor}
1856f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor
1857aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic enum CXVisitorResult findFileRefsVisit(void *context,
1858aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                         CXCursor cursor, CXSourceRange range) {
1859aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (clang_Range_isNull(range))
1860aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return CXVisit_Continue;
1861aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1862aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintCursor(cursor);
1863aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintRange(range, "");
1864aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  printf("\n");
1865aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  return CXVisit_Continue;
1866aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis}
1867aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1868aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidisstatic int find_file_refs_at(int argc, const char **argv) {
1869aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXIndex CIdx;
1870aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  int errorCode;
1871aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  struct CXUnsavedFile *unsaved_files = 0;
1872aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  int num_unsaved_files = 0;
1873aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXTranslationUnit TU;
1874aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXCursor Cursor;
1875aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CursorSourceLocation *Locations = 0;
1876aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned NumLocations = 0, Loc;
1877aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned Repeats = 1;
1878aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  unsigned I;
1879aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1880aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Count the number of locations. */
1881aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  while (strstr(argv[NumLocations+1], "-file-refs-at=") == argv[NumLocations+1])
1882aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    ++NumLocations;
1883aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1884aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Parse the locations. */
1885aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  assert(NumLocations > 0 && "Unable to count locations?");
1886aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  Locations = (CursorSourceLocation *)malloc(
1887aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  NumLocations * sizeof(CursorSourceLocation));
1888aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  for (Loc = 0; Loc < NumLocations; ++Loc) {
1889aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    const char *input = argv[Loc + 1] + strlen("-file-refs-at=");
1890aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if ((errorCode = parse_file_line_column(input, &Locations[Loc].filename,
1891aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                            &Locations[Loc].line,
1892aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                            &Locations[Loc].column, 0, 0)))
1893aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return errorCode;
1894aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
1895aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1896aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (parse_remapped_files(argc, argv, NumLocations + 1, &unsaved_files,
1897aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                           &num_unsaved_files))
1898aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return -1;
1899aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1900aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING"))
1901aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    Repeats = 5;
1902aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1903aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  /* Parse the translation unit. When we're testing clang_getCursor() after
1904aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis     reparsing, don't remap unsaved files until the second parse. */
1905aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CIdx = clang_createIndex(1, 1);
1906aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  TU = clang_parseTranslationUnit(CIdx, argv[argc - 1],
1907aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  argv + num_unsaved_files + 1 + NumLocations,
1908aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  argc - num_unsaved_files - 2 - NumLocations,
1909aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  unsaved_files,
1910aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  Repeats > 1? 0 : num_unsaved_files,
1911aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                  getDefaultParsingOptions());
1912aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1913aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (!TU) {
1914aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    fprintf(stderr, "unable to parse input\n");
1915aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return -1;
1916aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
1917aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1918dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0)
1919dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    return -1;
1920dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1921aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  for (I = 0; I != Repeats; ++I) {
1922aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if (Repeats > 1 &&
1923aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
1924aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                     clang_defaultReparseOptions(TU))) {
1925aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      clang_disposeTranslationUnit(TU);
1926aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return 1;
1927aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
1928dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1929dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    if (checkForErrors(TU) != 0)
1930dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      return -1;
1931aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1932aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    for (Loc = 0; Loc < NumLocations; ++Loc) {
1933aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      CXFile file = clang_getFile(TU, Locations[Loc].filename);
1934aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (!file)
1935aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        continue;
1936aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1937aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      Cursor = clang_getCursor(TU,
1938aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                               clang_getLocation(TU, file, Locations[Loc].line,
1939aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                                 Locations[Loc].column));
1940dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1941dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis      if (checkForErrors(TU) != 0)
1942dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        return -1;
1943dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1944aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (I + 1 == Repeats) {
194526fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen        CXCursorAndRangeVisitor visitor = { 0, findFileRefsVisit };
1946aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        PrintCursor(Cursor);
1947aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        printf("\n");
1948aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        clang_findReferencesInFile(Cursor, file, visitor);
1949aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        free(Locations[Loc].filename);
1950dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
1951dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis        if (checkForErrors(TU) != 0)
1952dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis          return -1;
1953aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      }
1954aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
1955aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
1956aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
1957aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  PrintDiagnostics(TU);
1958aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  clang_disposeTranslationUnit(TU);
1959aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  clang_disposeIndex(CIdx);
1960aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  free(Locations);
1961aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  free_remapped_files(unsaved_files, num_unsaved_files);
1962aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  return 0;
1963aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis}
1964aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis
19654e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidistypedef struct {
19664e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *check_prefix;
19674e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  int first_check_printed;
1968dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  int fail_for_error;
19696f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  int abort;
197013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  const char *main_filename;
19714e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis} IndexData;
19724e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
19734e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void printCheck(IndexData *data) {
19744e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (data->check_prefix) {
19754e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    if (data->first_check_printed) {
19764e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      printf("// %s-NEXT: ", data->check_prefix);
19774e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    } else {
19784e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      printf("// %s     : ", data->check_prefix);
19794e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      data->first_check_printed = 1;
19804e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    }
19814e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
19824e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
19834e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
1984dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic void printCXIndexFile(CXIdxClientFile file) {
19854e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString filename = clang_getFileName((CXFile)file);
19864e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("%s", clang_getCString(filename));
19874e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_disposeString(filename);
19884e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
19894e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
199013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidisstatic void printCXIndexLoc(CXIdxLoc loc, CXClientData client_data) {
199113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  IndexData *index_data;
19924e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString filename;
199313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  const char *cname;
1994dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  CXIdxClientFile file;
19954e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  unsigned line, column;
199613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  int isMainFile;
19974e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
199813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
19994e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_indexLoc_getFileLocation(loc, &file, 0, &line, &column, 0);
20004e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (line == 0) {
20014e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printf("<null loc>");
20024e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    return;
20034e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
2004c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  if (!file) {
2005c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis    printf("<no idxfile>");
2006c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis    return;
2007c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  }
20084e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  filename = clang_getFileName((CXFile)file);
20094e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  cname = clang_getCString(filename);
201013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  if (strcmp(cname, index_data->main_filename) == 0)
201113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    isMainFile = 1;
201213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  else
201313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    isMainFile = 0;
201413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  clang_disposeString(filename);
201513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
201613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  if (!isMainFile) {
20174e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printCXIndexFile(file);
20184e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    printf(":");
20194e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
20204e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("%d:%d", line, column);
20214e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
20224e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
202313c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidisstatic unsigned digitCount(unsigned val) {
202413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  unsigned c = 1;
202513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  while (1) {
202613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    if (val < 10)
202713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis      return c;
202813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    ++c;
202913c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    val /= 10;
203013c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  }
203113c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis}
203213c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
20336ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidisstatic CXIdxClientContainer makeClientContainer(const CXIdxEntityInfo *info,
20346ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                                CXIdxLoc loc) {
20354e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *name;
20364e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  char *newStr;
2037dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  CXIdxClientFile file;
20384e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  unsigned line, column;
20394e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2040dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  name = info->name;
20414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (!name)
20424e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    name = "<anon-tag>";
20434e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
20444e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  clang_indexLoc_getFileLocation(loc, &file, 0, &line, &column, 0);
2045f89bc0500c74a590ff9e99aab75bcc22d05bc760Argyrios Kyrtzidis  /* FIXME: free these.*/
204613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  newStr = (char *)malloc(strlen(name) +
204713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis                          digitCount(line) + digitCount(column) + 3);
20484e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  sprintf(newStr, "%s:%d:%d", name, line, column);
20496ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  return (CXIdxClientContainer)newStr;
20504e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
20514e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
20522957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidisstatic void printCXIndexContainer(const CXIdxContainerInfo *info) {
20532957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIdxClientContainer container;
20542957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  container = clang_index_getClientContainer(info);
20553e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis  if (!container)
20563e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis    printf("[<<NULL>>]");
20573e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis  else
20583e340a67f4522a2f633a719cb7f3389fe2474374Argyrios Kyrtzidis    printf("[%s]", (const char *)container);
20594e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
20604e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2061dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic const char *getEntityKindString(CXIdxEntityKind kind) {
2062dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  switch (kind) {
2063dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Unexposed: return "<<UNEXPOSED>>";
2064dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Typedef: return "typedef";
2065dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Function: return "function";
2066dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Variable: return "variable";
2067dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Field: return "field";
2068dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_EnumConstant: return "enumerator";
2069dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCClass: return "objc-class";
2070dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCProtocol: return "objc-protocol";
2071dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCCategory: return "objc-category";
2072c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  case CXIdxEntity_ObjCInstanceMethod: return "objc-instance-method";
2073c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  case CXIdxEntity_ObjCClassMethod: return "objc-class-method";
2074dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCProperty: return "objc-property";
2075dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_ObjCIvar: return "objc-ivar";
2076dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Enum: return "enum";
2077dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Struct: return "struct";
2078dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_Union: return "union";
2079dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  case CXIdxEntity_CXXClass: return "c++-class";
20802957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXNamespace: return "namespace";
20812957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXNamespaceAlias: return "namespace-alias";
20822957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXStaticVariable: return "c++-static-var";
20832957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXStaticMethod: return "c++-static-method";
20842957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXInstanceMethod: return "c++-instance-method";
20852957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXConstructor: return "constructor";
20862957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXDestructor: return "destructor";
20872957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXConversionFunction: return "conversion-func";
20882957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_CXXTypeAlias: return "type-alias";
20892957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  }
20902957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  assert(0 && "Garbage entity kind");
20912957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  return 0;
20922957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis}
20932957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis
20942957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidisstatic const char *getEntityTemplateKindString(CXIdxEntityCXXTemplateKind kind) {
20952957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  switch (kind) {
20962957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_NonTemplate: return "";
20972957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_Template: return "-template";
20982957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_TemplatePartialSpecialization:
20992957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    return "-template-partial-spec";
21002957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  case CXIdxEntity_TemplateSpecialization: return "-template-spec";
2101dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
21026ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  assert(0 && "Garbage entity kind");
21036ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  return 0;
2104dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis}
2105dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2106838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidisstatic const char *getEntityLanguageString(CXIdxEntityLanguage kind) {
2107838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  switch (kind) {
2108838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_None: return "<none>";
2109838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_C: return "C";
2110838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_ObjC: return "ObjC";
2111838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  case CXIdxEntityLang_CXX: return "C++";
2112838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  }
2113838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  assert(0 && "Garbage language kind");
2114838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis  return 0;
2115838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis}
2116838d3c23204f52ae27a9f5e9a254238a7ac5d41bArgyrios Kyrtzidis
2117dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic void printEntityInfo(const char *cb,
2118dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                            CXClientData client_data,
21196ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                            const CXIdxEntityInfo *info) {
21204e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *name;
21214e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
2122643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  unsigned i;
21234e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
21244e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
21254e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2126c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis  if (!info) {
2127c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis    printf("%s: <<NULL>>", cb);
2128c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis    return;
2129c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis  }
2130c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis
2131dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  name = info->name;
21324e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (!name)
21334e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    name = "<anon-tag>";
21344e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
21352957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  printf("%s: kind: %s%s", cb, getEntityKindString(info->kind),
21362957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis         getEntityTemplateKindString(info->templateKind));
2137dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | name: %s", name);
2138dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | USR: %s", info->USR);
2139c2be04eaec94e20fc825fb98b713112d9d82562fArgyrios Kyrtzidis  printf(" | lang: %s", getEntityLanguageString(info->lang));
2140643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis
2141643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  for (i = 0; i != info->numAttributes; ++i) {
2142643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis    const CXIdxAttrInfo *Attr = info->attributes[i];
2143643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis    printf("     <attribute>: ");
2144643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis    PrintCursor(Attr->cursor);
2145643d3ce93c501d19353f2fa578fee3e97f1d1b4bArgyrios Kyrtzidis  }
21464e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
21474e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2148b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidisstatic void printBaseClassInfo(CXClientData client_data,
2149b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis                               const CXIdxBaseClassInfo *info) {
2150b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printEntityInfo("     <base>", client_data, info->base);
2151b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printf(" | cursor: ");
2152b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  PrintCursor(info->cursor);
2153b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  printf(" | loc: ");
215413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2155b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis}
2156b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis
2157c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidisstatic void printProtocolList(const CXIdxObjCProtocolRefListInfo *ProtoInfo,
2158c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis                              CXClientData client_data) {
2159c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  unsigned i;
2160c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  for (i = 0; i < ProtoInfo->numProtocols; ++i) {
2161c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printEntityInfo("     <protocol>", client_data,
2162c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis                    ProtoInfo->protocols[i]->protocol);
2163c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf(" | cursor: ");
2164c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    PrintCursor(ProtoInfo->protocols[i]->cursor);
2165c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf(" | loc: ");
216613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    printCXIndexLoc(ProtoInfo->protocols[i]->loc, client_data);
2167c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printf("\n");
2168c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  }
2169c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis}
2170c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis
21714e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void index_diagnostic(CXClientData client_data,
2172996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis                             CXDiagnosticSet diagSet, void *reserved) {
21734e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  CXString str;
21744e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *cstr;
2175996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  unsigned numDiags, i;
2176996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  CXDiagnostic diag;
21774e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
21784e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
21794e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
21804e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2181996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  numDiags = clang_getNumDiagnosticsInSet(diagSet);
2182996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis  for (i = 0; i != numDiags; ++i) {
2183996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    diag = clang_getDiagnosticInSet(diagSet, i);
2184996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    str = clang_formatDiagnostic(diag, clang_defaultDiagnosticDisplayOptions());
2185996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    cstr = clang_getCString(str);
2186996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    printf("[diagnostic]: %s\n", cstr);
2187996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    clang_disposeString(str);
2188996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis
2189996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    if (getenv("CINDEXTEST_FAILONERROR") &&
2190996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis        clang_getDiagnosticSeverity(diag) >= CXDiagnostic_Error) {
2191996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis      index_data->fail_for_error = 1;
2192996e6e564af7483e2d5e0b70df5fdb9f79ec4b5aArgyrios Kyrtzidis    }
2193dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
21944e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
21954e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2196dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientFile index_enteredMainFile(CXClientData client_data,
2197dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                                       CXFile file, void *reserved) {
2198dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  IndexData *index_data;
219962d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis  CXString filename;
220062d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis
2201dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
2202dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCheck(index_data);
2203dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
220462d7fea624f466d11001b2dc733bee12a3b90c3aArgyrios Kyrtzidis  filename = clang_getFileName(file);
220513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  index_data->main_filename = clang_getCString(filename);
220613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  clang_disposeString(filename);
220713c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis
2208dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf("[enteredMainFile]: ");
2209dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)file);
2210dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf("\n");
2211dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2212dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientFile)file;
22134e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
22144e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2215dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientFile index_ppIncludedFile(CXClientData client_data,
22166ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                            const CXIdxIncludedFileInfo *info) {
22174e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
22184e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
22194e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
22204e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
222166042b32b3b37ddcba731ff05c2792e3bb572102Argyrios Kyrtzidis  printf("[ppIncludedFile]: ");
2222dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printCXIndexFile((CXIdxClientFile)info->file);
22234e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | name: \"%s\"", info->filename);
22244e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | hash loc: ");
222513c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->hashLoc, client_data);
22264e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | isImport: %d | isAngled: %d\n", info->isImport, info->isAngled);
2227dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis
2228dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientFile)info->file;
22294e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
22304e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2231dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidisstatic CXIdxClientContainer index_startedTranslationUnit(CXClientData client_data,
22324e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis                                                   void *reserved) {
22334e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData *index_data;
22344e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data = (IndexData *)client_data;
22354e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCheck(index_data);
22364e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
223766042b32b3b37ddcba731ff05c2792e3bb572102Argyrios Kyrtzidis  printf("[startedTranslationUnit]\n");
2238dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  return (CXIdxClientContainer)"TU";
22394e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
22404e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
22416ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidisstatic void index_indexDeclaration(CXClientData client_data,
22422957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                                   const CXIdxDeclInfo *info) {
2243dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  IndexData *index_data;
22446ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  const CXIdxObjCCategoryDeclInfo *CatInfo;
22456ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  const CXIdxObjCInterfaceDeclInfo *InterInfo;
2246c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  const CXIdxObjCProtocolRefListInfo *ProtoInfo;
2247792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  const CXIdxObjCPropertyDeclInfo *PropInfo;
2248b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  const CXIdxCXXClassDeclInfo *CXXClassInfo;
2249b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  unsigned i;
2250dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
22514e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2252dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo("[indexDeclaration]", client_data, info->entityInfo);
2253dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | cursor: ");
2254dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  PrintCursor(info->cursor);
2255dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | loc: ");
225613c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2257b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printf(" | semantic-container: ");
2258b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printCXIndexContainer(info->semanticContainer);
2259b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printf(" | lexical-container: ");
2260b1febb646bf7a2f319ad894c9833968c52d21711Argyrios Kyrtzidis  printCXIndexContainer(info->lexicalContainer);
2261dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printf(" | isRedecl: %d", info->isRedeclaration);
2262c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isDef: %d", info->isDefinition);
2263c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isContainer: %d", info->isContainer);
2264c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | isImplicit: %d\n", info->isImplicit);
22654e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2266b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  for (i = 0; i != info->numAttributes; ++i) {
226787adb0bf9375390de4c66d9e2ad110cc492cd655NAKAMURA Takumi    const CXIdxAttrInfo *Attr = info->attributes[i];
2268b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis    printf("     <attribute>: ");
2269b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis    PrintCursor(Attr->cursor);
2270b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis    printf("\n");
2271b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  }
2272b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis
2273dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  if (clang_index_isEntityObjCContainerKind(info->entityInfo->kind)) {
2274dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    const char *kindName = 0;
2275dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    CXIdxObjCContainerKind K = clang_index_getObjCContainerDeclInfo(info)->kind;
2276dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    switch (K) {
2277dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_ForwardRef:
2278dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "forward-ref"; break;
2279dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_Interface:
2280dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "interface"; break;
2281dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    case CXIdxObjCContainer_Implementation:
2282dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis      kindName = "implementation"; break;
2283dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    }
2284dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printCheck(index_data);
2285dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printf("     <ObjCContainerInfo>: kind: %s\n", kindName);
2286dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
22874e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
22886ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  if ((CatInfo = clang_index_getObjCCategoryDeclInfo(info))) {
2289dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printEntityInfo("     <ObjCCategoryInfo>: class", client_data,
2290dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis                    CatInfo->objcClass);
229121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    printf(" | cursor: ");
229221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    PrintCursor(CatInfo->classCursor);
229321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    printf(" | loc: ");
229413c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis    printCXIndexLoc(CatInfo->classLoc, client_data);
2295dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis    printf("\n");
2296dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  }
22974e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
22986ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  if ((InterInfo = clang_index_getObjCInterfaceDeclInfo(info))) {
22996ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis    if (InterInfo->superInfo) {
2300b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printBaseClassInfo(client_data, InterInfo->superInfo);
23016ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis      printf("\n");
23026ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis    }
23034e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
23044e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2305c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  if ((ProtoInfo = clang_index_getObjCProtocolRefListInfo(info))) {
2306c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis    printProtocolList(ProtoInfo, client_data);
23076ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis  }
23084e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2309792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  if ((PropInfo = clang_index_getObjCPropertyDeclInfo(info))) {
2310792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    if (PropInfo->getter) {
2311792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printEntityInfo("     <getter>", client_data, PropInfo->getter);
2312792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printf("\n");
2313792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    }
2314792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    if (PropInfo->setter) {
2315792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printEntityInfo("     <setter>", client_data, PropInfo->setter);
2316792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis      printf("\n");
2317792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis    }
2318792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis  }
2319792db266f3d2f12a7a16bf37d90074f54bca1e6fArgyrios Kyrtzidis
2320b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  if ((CXXClassInfo = clang_index_getCXXClassDeclInfo(info))) {
2321b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis    for (i = 0; i != CXXClassInfo->numBases; ++i) {
2322b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printBaseClassInfo(client_data, CXXClassInfo->bases[i]);
2323b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis      printf("\n");
2324b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis    }
2325b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis  }
2326b526a871af40b84d9878eded54a181bf4003b376Argyrios Kyrtzidis
23272957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  if (info->declAsContainer)
23282957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    clang_index_setClientContainer(info->declAsContainer,
23292957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                              makeClientContainer(info->entityInfo, info->loc));
23304e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
23314e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
23324e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic void index_indexEntityReference(CXClientData client_data,
23336ec43adc39006a7fce94188956d0239bd54c0363Argyrios Kyrtzidis                                       const CXIdxEntityRefInfo *info) {
2334dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo("[indexEntityReference]", client_data, info->referencedEntity);
23354e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | cursor: ");
23364e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  PrintCursor(info->cursor);
23374e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | loc: ");
233813c20a77d486c1ce0b6db2b836f2d14daa9e95ffArgyrios Kyrtzidis  printCXIndexLoc(info->loc, client_data);
2339dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  printEntityInfo(" | <parent>:", client_data, info->parentEntity);
23404e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf(" | container: ");
23414e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printCXIndexContainer(info->container);
2342c71d55469e7d5f7b376a30620617a175a9442da9Argyrios Kyrtzidis  printf(" | refkind: ");
2343aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  switch (info->kind) {
2344aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  case CXIdxEntityRef_Direct: printf("direct"); break;
2345b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  case CXIdxEntityRef_Implicit: printf("implicit"); break;
2346aca19be8731fc31cff68702de0dc7f30ce908979Argyrios Kyrtzidis  }
23474e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  printf("\n");
23484e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
23494e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
23506f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidisstatic int index_abortQuery(CXClientData client_data, void *reserved) {
23516f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  IndexData *index_data;
23526f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_data = (IndexData *)client_data;
23536f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  return index_data->abort;
23546f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis}
23556f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis
23564e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic IndexerCallbacks IndexCB = {
23576f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_abortQuery,
23584e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_diagnostic,
2359dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_enteredMainFile,
23604e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_ppIncludedFile,
2361f89bc0500c74a590ff9e99aab75bcc22d05bc760Argyrios Kyrtzidis  0, /*importedASTFile*/
23624e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_startedTranslationUnit,
2363dd93c596cd95e1b96031ff47efe0a5095ff3d7f1Argyrios Kyrtzidis  index_indexDeclaration,
23644e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_indexEntityReference
23654e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis};
23664e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
236722490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidisstatic unsigned getIndexOptions(void) {
236822490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  unsigned index_opts;
236922490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  index_opts = 0;
237022490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  if (getenv("CINDEXTEST_SUPPRESSREFS"))
237122490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis    index_opts |= CXIndexOpt_SuppressRedundantRefs;
237222490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  if (getenv("CINDEXTEST_INDEXLOCALSYMBOLS"))
237322490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis    index_opts |= CXIndexOpt_IndexFunctionLocalSymbols;
237422490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis
237522490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  return index_opts;
237622490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis}
237722490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis
23784e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidisstatic int index_file(int argc, const char **argv) {
23794e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  const char *check_prefix;
23802957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndex Idx;
23812957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndexAction idxAction;
23824e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  IndexData index_data;
2383b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  unsigned index_opts;
2384dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  int result;
23854e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
23864e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  check_prefix = 0;
23874e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc > 0) {
23884e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
23894e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
23904e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      ++argv;
23914e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis      --argc;
23924e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    }
23934e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
23944e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
23954e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc == 0) {
23964e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    fprintf(stderr, "no compiler arguments\n");
23974e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    return -1;
23984e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  }
23994e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
24002957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
24012957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                                /* displayDiagnosics=*/1))) {
24022957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
24032957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    return 1;
24042957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  }
24052957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  idxAction = 0;
24062957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis
24074e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data.check_prefix = check_prefix;
24084e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  index_data.first_check_printed = 0;
2409dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  index_data.fail_for_error = 0;
24106f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_data.abort = 0;
2411dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
241222490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  index_opts = getIndexOptions();
24132957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
24142957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  result = clang_indexSourceFile(idxAction, &index_data,
2415b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis                                 &IndexCB,sizeof(IndexCB), index_opts,
2416c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis                                 0, argv, argc, 0, 0, 0, 0);
241721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (index_data.fail_for_error)
24182957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    result = -1;
241921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
24202957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
24212957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_disposeIndex(Idx);
242221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  return result;
242321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis}
242421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
242521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidisstatic int index_tu(int argc, const char **argv) {
242621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  CXIndex Idx;
24272957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  CXIndexAction idxAction;
242821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  CXTranslationUnit TU;
242921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  const char *check_prefix;
243021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  IndexData index_data;
2431b395c63b473bf1b3783bff371a993332e8c4c5e3Argyrios Kyrtzidis  unsigned index_opts;
243221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  int result;
243321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
243421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  check_prefix = 0;
243521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc > 0) {
243621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    if (strstr(argv[0], "-check-prefix=") == argv[0]) {
243721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      check_prefix = argv[0] + strlen("-check-prefix=");
243821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      ++argv;
243921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis      --argc;
244021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    }
244121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
244221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
244321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc == 0) {
244421ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    fprintf(stderr, "no ast file\n");
244521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return -1;
244621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
244721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
244821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (!(Idx = clang_createIndex(/* excludeDeclsFromPCH */ 1,
244921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis                                /* displayDiagnosics=*/1))) {
245021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    fprintf(stderr, "Could not create Index\n");
245121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return 1;
245221ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  }
24532957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  idxAction = 0;
24542957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  result = 1;
245521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
245621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (!CreateTranslationUnit(Idx, argv[0], &TU))
24572957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    goto finished;
245821ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
245921ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  index_data.check_prefix = check_prefix;
246021ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  index_data.first_check_printed = 0;
246121ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  index_data.fail_for_error = 0;
24626f3ce979a7748fd117e6473d6272b16d643b6262Argyrios Kyrtzidis  index_data.abort = 0;
246321ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis
246422490747c123a78feec089539f30426084d348cfArgyrios Kyrtzidis  index_opts = getIndexOptions();
24652957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  idxAction = clang_IndexAction_create(Idx);
24662957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  result = clang_indexTranslationUnit(idxAction, &index_data,
2467c6b4a5099588fd21b49c80f730a596a64b2766c6Argyrios Kyrtzidis                                      &IndexCB,sizeof(IndexCB),
24682957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis                                      index_opts, TU);
2469dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (index_data.fail_for_error)
24702957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis    goto finished;
24714e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
24722957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  finished:
24732957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_IndexAction_dispose(idxAction);
24742957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis  clang_disposeIndex(Idx);
24752957e6f8c4c2e58a4b9cb639949fea801970fe36Argyrios Kyrtzidis
2476dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  return result;
24774e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
24784e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
2479fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregorint perform_token_annotation(int argc, const char **argv) {
2480fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  const char *input = argv[1];
2481fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  char *filename = 0;
2482fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned line, second_line;
2483fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned column, second_column;
2484fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXIndex CIdx;
2485fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXTranslationUnit TU = 0;
2486fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  int errorCode;
2487fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
2488fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  int num_unsaved_files = 0;
2489fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXToken *tokens;
2490fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned num_tokens;
2491fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXSourceRange range;
2492fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXSourceLocation startLoc, endLoc;
2493fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXFile file = 0;
2494fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  CXCursor *cursors = 0;
2495fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  unsigned i;
2496fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2497fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  input += strlen("-test-annotate-tokens=");
2498fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if ((errorCode = parse_file_line_column(input, &filename, &line, &column,
2499fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                                          &second_line, &second_column)))
2500fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return errorCode;
2501fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2502e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  if (parse_remapped_files(argc, argv, 2, &unsaved_files, &num_unsaved_files)) {
2503e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    free(filename);
2504fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return -1;
2505e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  }
2506fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
25070a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  CIdx = clang_createIndex(0, 1);
2508dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor  TU = clang_parseTranslationUnit(CIdx, argv[argc - 1],
2509dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  argv + num_unsaved_files + 2,
2510dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  argc - num_unsaved_files - 3,
2511dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  unsaved_files,
2512dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  num_unsaved_files,
2513dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                  getDefaultParsingOptions());
2514fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!TU) {
2515fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    fprintf(stderr, "unable to parse input\n");
2516fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    clang_disposeIndex(CIdx);
2517fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    free(filename);
2518fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
2519fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return -1;
2520e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  }
2521fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  errorCode = 0;
2522fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2523e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  if (checkForErrors(TU) != 0) {
2524e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    errorCode = -1;
2525e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith    goto teardown;
2526e07c5f897e8da88959c93a9d98f1b441da649eb6Richard Smith  }
2527dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2528ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (getenv("CINDEXTEST_EDITING")) {
2529ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    for (i = 0; i < 5; ++i) {
2530ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      if (clang_reparseTranslationUnit(TU, num_unsaved_files, unsaved_files,
2531ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                                       clang_defaultReparseOptions(TU))) {
2532ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        fprintf(stderr, "Unable to reparse translation unit!\n");
2533ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        errorCode = -1;
2534ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        goto teardown;
2535ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      }
2536ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    }
2537ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
2538ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
2539dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
2540dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
2541dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
2542dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
2543dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2544fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  file = clang_getFile(TU, filename);
2545fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!file) {
2546fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    fprintf(stderr, "file %s is not in this translation unit\n", filename);
2547fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
2548fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    goto teardown;
2549fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
2550fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2551fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  startLoc = clang_getLocation(TU, file, line, column);
2552fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (clang_equalLocations(clang_getNullLocation(), startLoc)) {
2553e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(stderr, "invalid source location %s:%d:%d\n", filename, line,
2554fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            column);
2555fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
2556e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    goto teardown;
2557fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
2558fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2559fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  endLoc = clang_getLocation(TU, file, second_line, second_column);
2560fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (clang_equalLocations(clang_getNullLocation(), endLoc)) {
2561e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    fprintf(stderr, "invalid source location %s:%d:%d\n", filename,
2562fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            second_line, second_column);
2563fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    errorCode = -1;
2564e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek    goto teardown;
2565fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
2566fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2567fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  range = clang_getRange(startLoc, endLoc);
2568fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_tokenize(TU, range, &tokens, &num_tokens);
2569dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2570dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
2571dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
2572dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
2573dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
2574dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2575fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  cursors = (CXCursor *)malloc(num_tokens * sizeof(CXCursor));
2576fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_annotateTokens(TU, tokens, num_tokens, cursors);
2577dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2578dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  if (checkForErrors(TU) != 0) {
2579dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    errorCode = -1;
2580dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis    goto teardown;
2581dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis  }
2582dfca64ddfbdbc48085119f7c943d05881899bf5eArgyrios Kyrtzidis
2583fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  for (i = 0; i != num_tokens; ++i) {
2584fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *kind = "<unknown>";
2585fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXString spelling = clang_getTokenSpelling(TU, tokens[i]);
2586fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXSourceRange extent = clang_getTokenExtent(TU, tokens[i]);
2587fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    unsigned start_line, start_column, end_line, end_column;
2588fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2589fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    switch (clang_getTokenKind(tokens[i])) {
2590fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Punctuation: kind = "Punctuation"; break;
2591fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Keyword: kind = "Keyword"; break;
2592fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Identifier: kind = "Identifier"; break;
2593fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Literal: kind = "Literal"; break;
2594fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    case CXToken_Comment: kind = "Comment"; break;
2595fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
2596a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(clang_getRangeStart(extent),
2597a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              0, &start_line, &start_column, 0);
2598a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    clang_getSpellingLocation(clang_getRangeEnd(extent),
2599a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                              0, &end_line, &end_column, 0);
260051b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar    printf("%s: \"%s\" ", kind, clang_getCString(spelling));
2601342742adb69af8ef8f2c652f162496b98c45bc20Benjamin Kramer    clang_disposeString(spelling);
260251b058cb1e726c49fe0fae29404a4ca4308a6a12Daniel Dunbar    PrintExtent(stdout, start_line, start_column, end_line, end_column);
26030045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    if (!clang_isInvalid(cursors[i].kind)) {
26040045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor      printf(" ");
2605aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      PrintCursor(cursors[i]);
26060045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    }
26070045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    printf("\n");
2608fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
2609fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free(cursors);
261093f5e6a5d7690f90bc8a94e6b40d6c7d19719e0cTed Kremenek  clang_disposeTokens(TU, tokens, num_tokens);
2611fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2612fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor teardown:
2613a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  PrintDiagnostics(TU);
2614fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_disposeTranslationUnit(TU);
2615fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  clang_disposeIndex(CIdx);
2616fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free(filename);
2617fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
2618fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  return errorCode;
2619fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
2620fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
2621db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonstatic int
2622db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisonperform_test_compilation_db(const char *database, int argc, const char **argv) {
2623db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompilationDatabase db;
2624db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompileCommands CCmds;
2625db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompileCommand CCmd;
2626db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXCompilationDatabase_Error ec;
2627db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXString wd;
2628db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  CXString arg;
2629db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  int errorCode = 0;
2630db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  char *tmp;
2631db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  unsigned len;
2632db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  char *buildDir;
2633db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  int i, j, a, numCmds, numArgs;
2634db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2635db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  len = strlen(database);
2636db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  tmp = (char *) malloc(len+1);
2637db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  memcpy(tmp, database, len+1);
2638db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  buildDir = dirname(tmp);
2639db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2640c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison  db = clang_CompilationDatabase_fromDirectory(buildDir, &ec);
2641db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2642db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  if (db) {
2643db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2644db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    if (ec!=CXCompilationDatabase_NoError) {
2645db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      printf("unexpected error %d code while loading compilation database\n", ec);
2646db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      errorCode = -1;
2647db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      goto cdb_end;
2648db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    }
2649db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2650db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    for (i=0; i<argc && errorCode==0; ) {
2651db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      if (strcmp(argv[i],"lookup")==0){
2652c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        CCmds = clang_CompilationDatabase_getCompileCommands(db, argv[i+1]);
2653db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2654db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (!CCmds) {
2655db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("file %s not found in compilation db\n", argv[i+1]);
2656db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          errorCode = -1;
2657db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          break;
2658db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
2659db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2660c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        numCmds = clang_CompileCommands_getSize(CCmds);
2661db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2662db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        if (numCmds==0) {
2663db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          fprintf(stderr, "should not get an empty compileCommand set for file"
2664db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison                          " '%s'\n", argv[i+1]);
2665db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          errorCode = -1;
2666db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          break;
2667db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
2668db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2669db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        for (j=0; j<numCmds; ++j) {
2670c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          CCmd = clang_CompileCommands_getCommand(CCmds, j);
2671db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2672c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          wd = clang_CompileCommand_getDirectory(CCmd);
2673db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("workdir:'%s'", clang_getCString(wd));
2674db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          clang_disposeString(wd);
2675db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2676db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf(" cmdline:'");
2677c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison          numArgs = clang_CompileCommand_getNumArgs(CCmd);
2678db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          for (a=0; a<numArgs; ++a) {
2679db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            if (a) printf(" ");
2680c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison            arg = clang_CompileCommand_getArg(CCmd, a);
2681db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            printf("%s", clang_getCString(arg));
2682db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison            clang_disposeString(arg);
2683db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          }
2684db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison          printf("'\n");
2685db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        }
2686db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2687c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison        clang_CompileCommands_dispose(CCmds);
2688db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2689db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison        i += 2;
2690db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison      }
2691db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    }
2692c70851b74c06b27df07b9a772656a2e663d432eeArnaud A. de Grandmaison    clang_CompilationDatabase_dispose(db);
2693db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  } else {
2694db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    printf("database loading failed with error code %d.\n", ec);
2695db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    errorCode = -1;
2696db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  }
2697db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2698db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaisoncdb_end:
2699db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  free(tmp);
2700db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
2701db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  return errorCode;
2702db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison}
2703db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
27040d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
2705f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/* USR printing.                                                              */
2706f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/******************************************************************************/
2707f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2708f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic int insufficient_usr(const char *kind, const char *usage) {
2709f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "USR for '%s' requires: %s\n", kind, usage);
2710f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 1;
2711f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2712f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2713f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic unsigned isUSR(const char *s) {
2714f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return s[0] == 'c' && s[1] == ':';
2715f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2716f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2717f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic int not_usr(const char *s, const char *arg) {
2718f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "'%s' argument ('%s') is not a USR\n", s, arg);
2719f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 1;
2720f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2721f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2722f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic void print_usr(CXString usr) {
2723f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  const char *s = clang_getCString(usr);
2724f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  printf("%s\n", s);
2725f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  clang_disposeString(usr);
2726f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2727f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2728f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekstatic void display_usrs() {
2729f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fprintf(stderr, "-print-usrs options:\n"
2730f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCCategory <class name> <category name>\n"
2731f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCClass <class name>\n"
2732f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCIvar <ivar name> <class USR>\n"
2733f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        " ObjCMethod <selector> [0=class method|1=instance method] "
2734f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            "<class USR>\n"
2735f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          " ObjCProperty <property name> <class USR>\n"
2736f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          " ObjCProtocol <protocol name>\n");
2737f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2738f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2739f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekint print_usrs(const char **I, const char **E) {
2740f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  while (I != E) {
2741f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    const char *kind = *I;
2742f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    unsigned len = strlen(kind);
2743f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    switch (len) {
2744f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 8:
2745f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCIvar", 8) == 0) {
2746f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
2747f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<ivar name> <class USR>");
2748f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[2]))
2749f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[2]);
2750f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
2751f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
2752a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[2];
2753ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
2754f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCIvar(I[1], x));
2755f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
2756f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2757f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
2758f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
2759f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
2760f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
2761f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 9:
2762f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCClass", 9) == 0) {
2763f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 1 >= E)
2764f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<class name>");
2765f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCClass(I[1]));
2766f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 2;
2767f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
2768f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
2769f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
2770f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 10:
2771f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCMethod", 10) == 0) {
2772f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 3 >= E)
2773f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<method selector> "
2774f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek                "[0=class method|1=instance method] <class USR>");
2775f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[3]))
2776f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[3]);
2777f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
2778f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
2779a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[3];
2780ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
2781f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCMethod(I[1], atoi(I[2]), x));
2782f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
2783f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 4;
2784f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
2785f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
2786f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
2787f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      case 12:
2788f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCCategory", 12) == 0) {
2789f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
2790f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<class name> <category name>");
2791f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCCategory(I[1], I[2]));
2792f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
2793f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
2794f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
2795f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCProtocol", 12) == 0) {
2796f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 1 >= E)
2797f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<protocol name>");
2798f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          print_usr(clang_constructUSR_ObjCProtocol(I[1]));
2799f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 2;
2800f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
2801f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
2802f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        if (memcmp(kind, "ObjCProperty", 12) == 0) {
2803f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (I + 2 >= E)
2804f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return insufficient_usr(kind, "<property name> <class USR>");
2805f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          if (!isUSR(I[2]))
2806f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            return not_usr("<class USR>", I[2]);
2807f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          else {
2808f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            CXString x;
2809a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            x.data = (void*) I[2];
2810ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek            x.private_flags = 0;
2811f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek            print_usr(clang_constructUSR_ObjCProperty(I[1], x));
2812f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          }
2813f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          I += 3;
2814f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek          continue;
2815f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        }
2816f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
2817f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      default:
2818f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        break;
2819f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
2820f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    break;
2821f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
2822f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2823f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  if (I != E) {
2824f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    fprintf(stderr, "Invalid USR kind: %s\n", *I);
2825f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    display_usrs();
2826f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return 1;
2827f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
2828f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 0;
2829f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2830f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2831f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenekint print_usrs_file(const char *file_name) {
2832f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  char line[2048];
2833f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  const char *args[128];
2834f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  unsigned numChars = 0;
2835f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2836f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  FILE *fp = fopen(file_name, "r");
2837f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  if (!fp) {
2838f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    fprintf(stderr, "error: cannot open '%s'\n", file_name);
2839f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return 1;
2840f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
2841f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2842f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  /* This code is not really all that safe, but it works fine for testing. */
2843f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  while (!feof(fp)) {
2844f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    char c = fgetc(fp);
2845f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    if (c == '\n') {
2846f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      unsigned i = 0;
2847f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      const char *s = 0;
2848f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2849f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (numChars == 0)
2850f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        continue;
2851f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2852f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      line[numChars] = '\0';
2853f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      numChars = 0;
2854f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2855f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (line[0] == '/' && line[1] == '/')
2856f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        continue;
2857f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2858f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      s = strtok(line, " ");
2859f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      while (s) {
2860f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        args[i] = s;
2861f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        ++i;
2862f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        s = strtok(0, " ");
2863f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      }
2864f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      if (print_usrs(&args[0], &args[i]))
2865f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek        return 1;
2866f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
2867f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    else
2868f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      line[numChars++] = c;
2869f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
2870f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2871f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  fclose(fp);
2872f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  return 0;
2873f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek}
2874f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek
2875f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek/******************************************************************************/
28760d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/* Command line processing.                                                   */
28770d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek/******************************************************************************/
28787ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregorint write_pch_file(const char *filename, int argc, const char *argv[]) {
28797ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  CXIndex Idx;
28807ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  CXTranslationUnit TU;
28817ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  struct CXUnsavedFile *unsaved_files = 0;
28827ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  int num_unsaved_files = 0;
288308aa622ab923fb8a2751f213feee0b9124ee86fbFrancois Pichet  int result = 0;
28847ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
28857ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  Idx = clang_createIndex(/* excludeDeclsFromPCH */1, /* displayDiagnosics=*/1);
28867ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
28877ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  if (parse_remapped_files(argc, argv, 0, &unsaved_files, &num_unsaved_files)) {
28887ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    clang_disposeIndex(Idx);
28897ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return -1;
28907ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  }
28917ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
28927ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  TU = clang_parseTranslationUnit(Idx, 0,
28937ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  argv + num_unsaved_files,
28947ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  argc - num_unsaved_files,
28957ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  unsaved_files,
28967ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  num_unsaved_files,
28977ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor                                  CXTranslationUnit_Incomplete);
28987ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  if (!TU) {
28997ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    fprintf(stderr, "Unable to load translation unit!\n");
29007ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    free_remapped_files(unsaved_files, num_unsaved_files);
29017ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    clang_disposeIndex(Idx);
29027ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return 1;
29037ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  }
29047ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
290539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  switch (clang_saveTranslationUnit(TU, filename,
290639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor                                    clang_defaultSaveOptions(TU))) {
290739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_None:
290839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
290939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
291039c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_TranslationErrors:
291139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: translation errors\n",
291239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor            filename);
291339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 2;
291439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
291539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
291639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_InvalidTU:
291739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: invalid translation unit\n",
291839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor            filename);
291939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 3;
292039c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
292139c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
292239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  case CXSaveError_Unknown:
292339c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  default:
292439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    fprintf(stderr, "Unable to write PCH file %s: unknown error \n", filename);
292539c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    result = 1;
292639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    break;
292739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  }
292839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor
29297ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  clang_disposeTranslationUnit(TU);
29307ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  free_remapped_files(unsaved_files, num_unsaved_files);
29317ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  clang_disposeIndex(Idx);
293239c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  return result;
29337ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor}
29347ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
29357ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/******************************************************************************/
2936153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/* Serialized diagnostics.                                                    */
2937153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/******************************************************************************/
2938153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
2939153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic const char *getDiagnosticCodeStr(enum CXLoadDiag_Error error) {
2940153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  switch (error) {
2941153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_CannotLoad: return "Cannot Load File";
2942153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_None: break;
2943153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_Unknown: return "Unknown";
2944153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXLoadDiag_InvalidFile: return "Invalid File";
2945153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
2946153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return "None";
2947153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
2948153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
2949153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic const char *getSeverityString(enum CXDiagnosticSeverity severity) {
2950153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  switch (severity) {
2951153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Note: return "note";
2952153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Error: return "error";
2953153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Fatal: return "fatal";
2954153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Ignored: return "ignored";
2955153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    case CXDiagnostic_Warning: return "warning";
2956153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
2957153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return "unknown";
2958153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
2959153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
2960153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printIndent(unsigned indent) {
2961a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  if (indent == 0)
2962a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek    return;
2963a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  fprintf(stderr, "+");
2964a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  --indent;
2965153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  while (indent > 0) {
2966a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek    fprintf(stderr, "-");
2967153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    --indent;
2968153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
2969153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
2970153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
2971153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printLocation(CXSourceLocation L) {
2972153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXFile File;
2973153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXString FileName;
2974153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned line, column, offset;
2975153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
2976153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_getExpansionLocation(L, &File, &line, &column, &offset);
2977153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  FileName = clang_getFileName(File);
2978153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
2979153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  fprintf(stderr, "%s:%d:%d", clang_getCString(FileName), line, column);
2980153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_disposeString(FileName);
2981153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
2982153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
2983153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printRanges(CXDiagnostic D, unsigned indent) {
2984153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned i, n = clang_getDiagnosticNumRanges(D);
2985153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
2986153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0; i < n; ++i) {
2987153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceLocation Start, End;
2988153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceRange SR = clang_getDiagnosticRange(D, i);
2989153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    Start = clang_getRangeStart(SR);
2990153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    End = clang_getRangeEnd(SR);
2991153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
2992153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
2993153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "Range: ");
2994153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(Start);
2995153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, " ");
2996153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(End);
2997153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "\n");
2998153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
2999153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3000153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3001153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printFixIts(CXDiagnostic D, unsigned indent) {
3002153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  unsigned i, n = clang_getDiagnosticNumFixIts(D);
30033739b32a01b9778176757425e62252c288b85383Ted Kremenek  fprintf(stderr, "Number FIXITs = %d\n", n);
3004153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0 ; i < n; ++i) {
3005153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceRange ReplacementRange;
3006153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXString text;
3007153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    text = clang_getDiagnosticFixIt(D, i, &ReplacementRange);
3008153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3009153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3010153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "FIXIT: (");
3011153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(clang_getRangeStart(ReplacementRange));
3012153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, " - ");
3013153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printLocation(clang_getRangeEnd(ReplacementRange));
3014153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "): \"%s\"\n", clang_getCString(text));
3015153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(text);
3016153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3017153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3018153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3019153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic void printDiagnosticSet(CXDiagnosticSet Diags, unsigned indent) {
30209190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi  unsigned i, n;
30219190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi
3022153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (!Diags)
3023153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    return;
3024153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
30259190943fef9584edd30d1861f3225c3a4d1c4f53NAKAMURA Takumi  n = clang_getNumDiagnosticsInSet(Diags);
3026153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  for (i = 0; i < n; ++i) {
3027153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXSourceLocation DiagLoc;
3028153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXDiagnostic D;
3029153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    CXFile File;
303078d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    CXString FileName, DiagSpelling, DiagOption, DiagCat;
3031153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    unsigned line, column, offset;
303278d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    const char *DiagOptionStr = 0, *DiagCatStr = 0;
3033153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3034153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    D = clang_getDiagnosticInSet(Diags, i);
3035153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagLoc = clang_getDiagnosticLocation(D);
3036153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_getExpansionLocation(DiagLoc, &File, &line, &column, &offset);
3037153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    FileName = clang_getFileName(File);
3038153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagSpelling = clang_getDiagnosticSpelling(D);
3039153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3040153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printIndent(indent);
3041153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3042153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "%s:%d:%d: %s: %s",
3043153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(FileName),
3044153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            line,
3045153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            column,
3046153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            getSeverityString(clang_getDiagnosticSeverity(D)),
3047153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(DiagSpelling));
3048153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3049153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagOption = clang_getDiagnosticOption(D, 0);
3050153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    DiagOptionStr = clang_getCString(DiagOption);
3051153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    if (DiagOptionStr) {
3052153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek      fprintf(stderr, " [%s]", DiagOptionStr);
3053153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    }
3054153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
305578d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    DiagCat = clang_getDiagnosticCategoryText(D);
305678d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    DiagCatStr = clang_getCString(DiagCat);
305778d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    if (DiagCatStr) {
305878d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek      fprintf(stderr, " [%s]", DiagCatStr);
305978d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek    }
306078d5d3bb9386a5aa31d19445eb8f81bf2652acb4Ted Kremenek
3061153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "\n");
3062153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3063153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printRanges(D, indent);
3064153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printFixIts(D, indent);
3065153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3066a4ca95a6ebe85d63df08f3290372e97ddaf0b02aNAKAMURA Takumi    /* Print subdiagnostics. */
3067153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    printDiagnosticSet(clang_getChildDiagnostics(D), indent+2);
3068153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3069153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(FileName);
3070153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(DiagSpelling);
3071153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(DiagOption);
3072153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3073153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3074153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3075153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenekstatic int read_diagnostics(const char *filename) {
3076153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  enum CXLoadDiag_Error error;
3077153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXString errorString;
3078153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  CXDiagnosticSet Diags = 0;
3079153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3080153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  Diags = clang_loadDiagnostics(filename, &error, &errorString);
3081153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (!Diags) {
3082153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    fprintf(stderr, "Trouble deserializing file (%s): %s\n",
3083153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            getDiagnosticCodeStr(error),
3084153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek            clang_getCString(errorString));
3085153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    clang_disposeString(errorString);
3086153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    return 1;
3087153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  }
3088153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3089153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  printDiagnosticSet(Diags, 0);
3090a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek  fprintf(stderr, "Number of diagnostics: %d\n",
3091a7e8a8394ccaa7a1276d32d0c16515bf4e7b486eTed Kremenek          clang_getNumDiagnosticsInSet(Diags));
3092153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  clang_disposeDiagnosticSet(Diags);
3093153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  return 0;
3094153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek}
3095153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
3096153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek/******************************************************************************/
30977ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/* Command line processing.                                                   */
30987ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor/******************************************************************************/
3099f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek
3100e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregorstatic CXCursorVisitor GetVisitor(const char *s) {
31017d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  if (s[0] == '\0')
3102e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    return FilteredPrintingVisitor;
31037d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  if (strcmp(s, "-usrs") == 0)
31047d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    return USRVisitor;
310559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  if (strncmp(s, "-memory-usage", 13) == 0)
310659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    return GetVisitor(s + 13);
31077d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  return NULL;
31087d40562f83552b7295411e10ee887d8d55470679Ted Kremenek}
31097d40562f83552b7295411e10ee887d8d55470679Ted Kremenek
3110f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenekstatic void print_usage(void) {
3111f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  fprintf(stderr,
31120d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "usage: c-index-test -code-completion-at=<site> <compiler arguments>\n"
31131982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -code-completion-timing=<site> <compiler arguments>\n"
3114f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    "       c-index-test -cursor-at=<site> <compiler arguments>\n"
3115aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    "       c-index-test -file-refs-at=<site> <compiler arguments>\n"
31164e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    "       c-index-test -index-file [-check-prefix=<FileCheck prefix>] <compiler arguments>\n"
311721ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    "       c-index-test -index-tu [-check-prefix=<FileCheck prefix>] <AST file>\n"
31181d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    "       c-index-test -test-file-scan <AST file> <source file> "
311926fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen          "[FileCheck prefix]\n");
312026fc0f9a078b1a9120547b36804f97c626817bdfErik Verbruggen  fprintf(stderr,
3121fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek    "       c-index-test -test-load-tu <AST file> <symbol filter> "
3122fe6fd3d41a7f48317d6856c9327b6cead32c3498Ted Kremenek          "[FileCheck prefix]\n"
31237d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    "       c-index-test -test-load-tu-usrs <AST file> <symbol filter> "
31247d40562f83552b7295411e10ee887d8d55470679Ted Kremenek           "[FileCheck prefix]\n"
31251982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -test-load-source <symbol filter> {<args>}*\n");
3126f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  fprintf(stderr,
312759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-load-source-memory-usage "
312859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "<symbol filter> {<args>}*\n"
3129abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    "       c-index-test -test-load-source-reparse <trials> <symbol filter> "
3130abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    "          {<args>}*\n"
31311982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    "       c-index-test -test-load-source-usrs <symbol filter> {<args>}*\n"
313259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-load-source-usrs-memory-usage "
313359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek          "<symbol filter> {<args>}*\n"
313416b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    "       c-index-test -test-annotate-tokens=<range> {<args>}*\n"
313516b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    "       c-index-test -test-inclusion-stack-source {<args>}*\n"
31364e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    "       c-index-test -test-inclusion-stack-tu <AST file>\n");
313753513d2e24205965ffd5d4780a10f41cc335f17bChandler Carruth  fprintf(stderr,
31384e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek    "       c-index-test -test-print-linkage-source {<args>}*\n"
313959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -test-print-typekind {<args>}*\n"
314059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    "       c-index-test -print-usr [<CursorKind> {<args>}]*\n"
31417ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    "       c-index-test -print-usr-file <file>\n"
3142153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    "       c-index-test -write-pch <file> <compiler arguments>\n");
3143153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  fprintf(stderr,
3144db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    "       c-index-test -compilation-db [lookup <filename>] database\n");
3145db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  fprintf(stderr,
3146153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    "       c-index-test -read-diagnostics <file>\n\n");
3147caf4bd3d3c66fd95b0159356e2c4da7fbf2897f3Douglas Gregor  fprintf(stderr,
31487d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    " <symbol filter> values:\n%s",
31490d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   all - load all symbols, including those from PCH\n"
31500d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   local - load all symbols except those in PCH\n"
31510d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   category - only load ObjC categories (non-PCH)\n"
31520d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   interface - only load ObjC interfaces (non-PCH)\n"
31530d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   protocol - only load ObjC protocols (non-PCH)\n"
31540d435191fc8f78338c2ef26d566cfe4d2c52c706Ted Kremenek    "   function - only load functions (non-PCH)\n"
3155625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    "   typedef - only load typdefs (non-PCH)\n"
3156625e4efc2c64b638285c9f836cc00a3ad7d0bd5fDaniel Dunbar    "   scan-function - scan function bodies (non-PCH)\n\n");
3157f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek}
3158f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek
31596edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/***/
31606edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
31616edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbarint cindextest_main(int argc, const char **argv) {
31620a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  clang_enableStackTraces();
3163153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  if (argc > 2 && strcmp(argv[1], "-read-diagnostics") == 0)
3164153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek      return read_diagnostics(argv[2]);
3165f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  if (argc > 2 && strstr(argv[1], "-code-completion-at=") == argv[1])
31661982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    return perform_code_completion(argc, argv, 0);
31671982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor  if (argc > 2 && strstr(argv[1], "-code-completion-timing=") == argv[1])
31681982c18522a4aefd57207bbd2d66d93945c41f92Douglas Gregor    return perform_code_completion(argc, argv, 1);
3169f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor  if (argc > 2 && strstr(argv[1], "-cursor-at=") == argv[1])
3170f2c87bd0368775927ead93e0dee3e4f3ca3f9a63Douglas Gregor    return inspect_cursor_at(argc, argv);
3171aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  if (argc > 2 && strstr(argv[1], "-file-refs-at=") == argv[1])
3172aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return find_file_refs_at(argc, argv);
31734e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-file") == 0)
31744e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    return index_file(argc - 2, argv + 2);
317521ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis  if (argc > 2 && strcmp(argv[1], "-index-tu") == 0)
317621ee5707e6e59d982d2f2ae28e079c7ff46dc519Argyrios Kyrtzidis    return index_tu(argc - 2, argv + 2);
31777d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strncmp(argv[1], "-test-load-tu", 13) == 0) {
3178e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 13);
31797d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    if (I)
3180ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek      return perform_test_load_tu(argv[2], argv[3], argc >= 5 ? argv[4] : 0, I,
3181ce2ae88f834c740a28b7e074a4477039918f9bb0Ted Kremenek                                  NULL);
31827d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  }
3183abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  else if (argc >= 5 && strncmp(argv[1], "-test-load-source-reparse", 25) == 0){
3184abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 25);
3185abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    if (I) {
3186abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      int trials = atoi(argv[2]);
3187abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor      return perform_test_reparse_source(argc - 4, argv + 4, trials, argv[3], I,
3188abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor                                         NULL);
3189abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    }
3190abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
31917d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strncmp(argv[1], "-test-load-source", 17) == 0) {
3192e5b72bac8312f83aeb2e29b3988ebfa30f2b4ca2Douglas Gregor    CXCursorVisitor I = GetVisitor(argv[1] + 17);
319359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
319459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    PostVisitTU postVisit = 0;
319559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    if (strstr(argv[1], "-memory-usage"))
319659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      postVisit = PrintMemoryUsage;
319759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
31987d40562f83552b7295411e10ee887d8d55470679Ted Kremenek    if (I)
319959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      return perform_test_load_source(argc - 3, argv + 3, argv[2], I,
320059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek                                      postVisit);
32017d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  }
32027d40562f83552b7295411e10ee887d8d55470679Ted Kremenek  else if (argc >= 4 && strcmp(argv[1], "-test-file-scan") == 0)
32031d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek    return perform_file_scan(argv[2], argv[3],
32041d5fdf3d3b5ea2640ebe8673814a0b6ab7cf5eb2Ted Kremenek                             argc >= 5 ? argv[4] : 0);
3205fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  else if (argc > 2 && strstr(argv[1], "-test-annotate-tokens=") == argv[1])
3206fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return perform_token_annotation(argc, argv);
320716b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-inclusion-stack-source") == 0)
320816b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all", NULL,
320916b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                                    PrintInclusionStack);
321016b55a71695a33c094383295cc7b7a2080e098daTed Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-inclusion-stack-tu") == 0)
321116b55a71695a33c094383295cc7b7a2080e098daTed Kremenek    return perform_test_load_tu(argv[2], "all", NULL, NULL,
321216b55a71695a33c094383295cc7b7a2080e098daTed Kremenek                                PrintInclusionStack);
32133bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-print-linkage-source") == 0)
32143bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all", PrintLinkage,
32153bed527b10502c6a5916ebb6e6abc49687ce1444Ted Kremenek                                    NULL);
32168e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-test-print-typekind") == 0)
32178e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek    return perform_test_load_source(argc - 2, argv + 2, "all",
32188e0ac174c8c8c980927b3e1447f16f62fbe2a2e4Ted Kremenek                                    PrintTypeKind, 0);
3219f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  else if (argc > 1 && strcmp(argv[1], "-print-usr") == 0) {
3220f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    if (argc > 2)
3221f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      return print_usrs(argv + 2, argv + argc);
3222f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    else {
3223f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      display_usrs();
3224f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek      return 1;
3225f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    }
3226f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  }
3227f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek  else if (argc > 2 && strcmp(argv[1], "-print-usr-file") == 0)
3228f7b714df46cdbdf9e2ebf26eb2fd7881790d83e6Ted Kremenek    return print_usrs_file(argv[2]);
32297ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  else if (argc > 2 && strcmp(argv[1], "-write-pch") == 0)
32307ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor    return write_pch_file(argv[2], argc - 3, argv + 3);
3231db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison  else if (argc > 2 && strcmp(argv[1], "-compilation-db") == 0)
3232db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison    return perform_test_compilation_db(argv[argc-1], argc - 3, argv + 2);
3233db29318326aba9e0c5c1bbba093687dccc38fc39Arnaud A. de Grandmaison
3234f5d9c9340b1a5ecd6ab62b916600b6e19e69539fTed Kremenek  print_usage();
323550398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff  return 1;
323650398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff}
32376edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
32386edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/***/
32396edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
32406edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar/* We intentionally run in a separate thread to ensure we at least minimal
32416edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar * testing of a multithreaded environment (for example, having a reduced stack
32426edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar * size). */
32436edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
32446edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbartypedef struct thread_info {
32456edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  int argc;
32466edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  const char **argv;
32476edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  int result;
32486edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar} thread_info;
324984294911b83003bcac726d00ffa91143752feb57Benjamin Kramervoid thread_runner(void *client_data_v) {
32506edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  thread_info *client_data = client_data_v;
32516edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data->result = cindextest_main(client_data->argc, client_data->argv);
32523be55cdbb5fda326dc55eb01c92d6645a2304309NAKAMURA Takumi#ifdef __CYGWIN__
32533be55cdbb5fda326dc55eb01c92d6645a2304309NAKAMURA Takumi  fflush(stdout);  /* stdout is not flushed on Cygwin. */
32543be55cdbb5fda326dc55eb01c92d6645a2304309NAKAMURA Takumi#endif
32556edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar}
32566edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
32576edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbarint main(int argc, const char **argv) {
32586edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  thread_info client_data;
32596edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar
32606160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor  if (getenv("CINDEXTEST_NOTHREADS"))
32616160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor    return cindextest_main(argc, argv);
32626160598bcd6549757525cbb463a551a0ad1615e4Douglas Gregor
32636edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data.argc = argc;
32646edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  client_data.argv = argv;
3265a32a6e1abe6972b48b4d5ec79d81b6f7241723bfDaniel Dunbar  clang_executeOnThread(thread_runner, &client_data, 0);
32666edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar  return client_data.result;
32676edc8001f2c2f5a95ed29259d8f2c0b9faa4bc83Daniel Dunbar}
3268