_Z14ImportCallbackPvPKcS1_PPcS3_Pm:
   26|     34|                     char** found_here, char **buf, size_t *buflen) {
   27|       |  // Don't load file and mark it as failure.
   28|       |  *buf = NULL;
   29|     34|  *buflen = 0;
   30|     34|  return 1;
   31|     34|}
_Z20ConvertJsonnetToJsonRKNSt3__112basic_stringIcNS_11char_traitsIcEENS_9allocatorIcEEEE:
   33|  7.42k|std::string ConvertJsonnetToJson(const std::string& jsonnet) {
   34|  7.42k|  JsonnetVm* jvm = jsonnet_make();
   35|  7.42k|  jsonnet_import_callback(jvm, ImportCallback, jvm);
   36|  7.42k|  int error = 0;
   37|  7.42k|  char* res =
   38|  7.42k|      jsonnet_evaluate_snippet_stream(jvm, /*filename=*/"", jsonnet.c_str(), &error);
   39|       |
   40|  7.42k|  std::string json;
   41|  7.42k|  if (error == 0 && res != nullptr) {
  ------------------
  |  Branch (41:7): [True: 508, False: 6.91k]
  |  Branch (41:21): [True: 508, False: 0]
  ------------------
   42|    508|    json = res;
   43|    508|  }
   44|       |
   45|  7.42k|  if (res) {
  ------------------
  |  Branch (45:7): [True: 7.42k, False: 0]
  ------------------
   46|  7.42k|    jsonnet_realloc(jvm, res, 0);
   47|  7.42k|  }
   48|  7.42k|  jsonnet_destroy(jvm);
   49|  7.42k|  return json;
   50|  7.42k|}
LLVMFuzzerTestOneInput:
   52|  7.42k|extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) {
   53|  7.42k|  std::string fuzz_jsonnet(reinterpret_cast<const char*>(data), size);
   54|  7.42k|  ConvertJsonnetToJson(fuzz_jsonnet);
   55|  7.42k|  return 0;
   56|  7.42k|}

libjsonnet.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
libjsonnet.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
libjsonnet.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}
_ZN7jsonnet8internal9AllocatorD2Ev:
 1020|  7.42k|    {
 1021|   104M|        for (auto x : allocated) {
  ------------------
  |  Branch (1021:21): [True: 104M, False: 7.42k]
  ------------------
 1022|   104M|            delete x;
 1023|   104M|        }
 1024|  7.42k|        allocated.clear();
 1025|  2.01M|        for (const auto &x : internedIdentifiers) {
  ------------------
  |  Branch (1025:28): [True: 2.01M, False: 7.42k]
  ------------------
 1026|  2.01M|            delete x.second;
 1027|  2.01M|        }
 1028|  7.42k|        internedIdentifiers.clear();
 1029|   172k|        for (const auto &x : internedBuiltins) {
  ------------------
  |  Branch (1029:28): [True: 172k, False: 7.42k]
  ------------------
 1030|   172k|            delete x.second;
 1031|   172k|        }
 1032|  7.42k|        internedBuiltins.clear();
 1033|  7.42k|    }
_ZN7jsonnet8internal10IdentifierC2ERKNSt3__112basic_stringIDiNS2_11char_traitsIDiEENS2_9allocatorIDiEEEE:
  113|  2.57M|    Identifier(const UString &name) : name(name) {}
_ZN7jsonnet8internal3ASTC2ERKNS0_13LocationRangeENS0_7ASTTypeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEE:
  132|  91.3M|        : location(location), type(type), openFodder(open_fodder)
  133|  91.3M|    {
  134|  91.3M|    }
_ZN7jsonnet8internal3ASTD2Ev:
  135|   104M|    virtual ~AST(void) {}
_ZN7jsonnet8internal8ArgParamC2ERKNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEEPKNS0_10IdentifierES9_:
  158|   599k|        : idFodder(id_fodder), id(id), expr(nullptr), commaFodder(comma_fodder)
  159|   599k|    {
  160|   599k|    }
_ZN7jsonnet8internal8ArgParamC2EPNS0_3ASTERKNSt3__16vectorINS0_13FodderElementENS4_9allocatorIS6_EEEE:
  163|  6.33M|        : id(nullptr), expr(expr), commaFodder(comma_fodder)
  164|  6.33M|    {
  165|  6.33M|    }
_ZN7jsonnet8internal8ArgParamC2ERKNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEEPKNS0_10IdentifierES9_PNS0_3ASTES9_:
  169|  13.4M|        : idFodder(id_fodder), id(id), eqFodder(eq_fodder), expr(expr), commaFodder(comma_fodder)
  170|  13.4M|    {
  171|  13.4M|    }
_ZN7jsonnet8internal17ComprehensionSpecC2ENS1_4KindERKNSt3__16vectorINS0_13FodderElementENS3_9allocatorIS5_EEEESA_PKNS0_10IdentifierESA_PNS0_3ASTE:
  187|   378k|        : kind(kind),
  188|   378k|          openFodder(open_fodder),
  189|   378k|          varFodder(var_fodder),
  190|   378k|          var(var),
  191|   378k|          inFodder(in_fodder),
  192|   378k|          expr(expr)
  193|   378k|    {
  194|   378k|    }
_ZN7jsonnet8internal5ApplyC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_RKNS6_INS0_8ArgParamENS8_ISF_EEEEbSC_SC_b:
  209|  9.47M|        : AST(lr, AST_APPLY, open_fodder),
  210|  9.47M|          target(target),
  211|  9.47M|          fodderL(fodder_l),
  212|  9.47M|          args(args),
  213|  9.47M|          trailingComma(trailing_comma),
  214|  9.47M|          fodderR(fodder_r),
  215|  9.47M|          tailstrictFodder(tailstrict_fodder),
  216|  9.47M|          tailstrict(tailstrict)
  217|  9.47M|    {
  218|  9.47M|    }
_ZN7jsonnet8internal10ApplyBraceC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESE_:
  226|  55.6k|        : AST(lr, AST_APPLY_BRACE, open_fodder), left(left), right(right)
  227|  55.6k|    {
  228|  55.6k|    }
_ZN7jsonnet8internal5Array7ElementC2EPNS0_3ASTERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEE:
  236|  2.30M|        Element(AST *expr, const Fodder &comma_fodder) : expr(expr), commaFodder(comma_fodder) {}
_ZN7jsonnet8internal5ArrayC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERKNS6_INS1_7ElementENS8_ISD_EEEEbSC_:
  244|  1.38M|        : AST(lr, AST_ARRAY, open_fodder),
  245|  1.38M|          elements(elements),
  246|  1.38M|          trailingComma(trailing_comma),
  247|  1.38M|          closeFodder(close_fodder)
  248|  1.38M|    {
  249|  1.38M|    }
_ZN7jsonnet8internal18ArrayComprehensionC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_bRKNS6_INS0_17ComprehensionSpecENS8_ISF_EEEESC_:
  262|   273k|        : AST(lr, AST_ARRAY_COMPREHENSION, open_fodder),
  263|   273k|          body(body),
  264|   273k|          commaFodder(comma_fodder),
  265|   273k|          trailingComma(trailing_comma),
  266|   273k|          specs(specs),
  267|   273k|          closeFodder(close_fodder)
  268|   273k|    {
  269|       |        assert(specs.size() > 0);
  ------------------
  |  Branch (269:9): [True: 273k, False: 0]
  ------------------
  270|   273k|    }
_ZN7jsonnet8internal6AssertC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_SE_SC_SE_:
  286|   310k|        : AST(lr, AST_ASSERT, open_fodder),
  287|   310k|          cond(cond),
  288|   310k|          colonFodder(colon_fodder),
  289|   310k|          message(message),
  290|   310k|          semicolonFodder(semicolon_fodder),
  291|   310k|          rest(rest)
  292|   310k|    {
  293|   310k|    }
_ZN7jsonnet8internal6BinaryC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_NS0_8BinaryOpESE_:
  367|  7.56M|        : AST(lr, AST_BINARY, open_fodder), left(left), opFodder(op_fodder), op(op), right(right)
  368|  7.56M|    {
  369|  7.56M|    }
_ZN7jsonnet8internal19BuiltinFunctionBodyC2ERKNS0_13LocationRangeERKNSt3__112basic_stringIcNS5_11char_traitsIcEENS5_9allocatorIcEEEERKNS5_6vectorIPKNS0_10IdentifierENS9_ISH_EEEE:
  382|   172k|        : AST(lr, AST_BUILTIN_FUNCTION_BODY, Fodder{}), name(name), params(params)
  383|   172k|    {
  384|   172k|    }
_ZN7jsonnet8internal15BuiltinFunctionC2ERKNS0_13LocationRangeEPKNS0_19BuiltinFunctionBodyE:
  395|   317k|        : AST(lr, AST_BUILTIN_FUNCTION, Fodder{}), body(body)
  396|   317k|    {
  397|   317k|    }
_ZN7jsonnet8internal11ConditionalC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_SE_SC_SE_:
  414|  3.50M|        : AST(lr, AST_CONDITIONAL, open_fodder),
  415|  3.50M|          cond(cond),
  416|  3.50M|          thenFodder(then_fodder),
  417|  3.50M|          branchTrue(branch_true),
  418|  3.50M|          elseFodder(else_fodder),
  419|  3.50M|          branchFalse(branch_false)
  420|  3.50M|    {
  421|  3.50M|    }
_ZN7jsonnet8internal6DollarC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEE:
  426|  26.0k|    Dollar(const LocationRange &lr, const Fodder &open_fodder) : AST(lr, AST_DOLLAR, open_fodder) {}
_ZN7jsonnet8internal5ErrorC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTE:
  433|  1.14M|        : AST(lr, AST_ERROR, open_fodder), expr(expr)
  434|  1.14M|    {
  435|  1.14M|    }
_ZN7jsonnet8internal8FunctionC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEESC_RKNS6_INS0_8ArgParamENS8_ISD_EEEEbSC_PNS0_3ASTE:
  448|  1.95M|        : AST(lr, AST_FUNCTION, open_fodder),
  449|  1.95M|          parenLeftFodder(paren_left_fodder),
  450|  1.95M|          params(params),
  451|  1.95M|          trailingComma(trailing_comma),
  452|  1.95M|          parenRightFodder(paren_right_fodder),
  453|  1.95M|          body(body)
  454|  1.95M|    {
  455|  1.95M|    }
_ZN7jsonnet8internal6ImportC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_13LiteralStringE:
  464|    729|        : AST(lr, AST_IMPORT, open_fodder), file(file)
  465|    729|    {
  466|    729|    }
_ZN7jsonnet8internal9ImportstrC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_13LiteralStringE:
  473|    570|        : AST(lr, AST_IMPORTSTR, open_fodder), file(file)
  474|    570|    {
  475|    570|    }
_ZN7jsonnet8internal9ImportbinC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_13LiteralStringE:
  482|    417|        : AST(lr, AST_IMPORTBIN, open_fodder), file(file)
  483|    417|    {
  484|    417|    }
_ZN7jsonnet8internal5IndexC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_SC_PKNS0_10IdentifierE:
  505|  5.43M|        : AST(lr, AST_INDEX, open_fodder),
  506|  5.43M|          target(target),
  507|  5.43M|          dotFodder(dot_fodder),
  508|  5.43M|          isSlice(false),
  509|  5.43M|          index(nullptr),
  510|  5.43M|          end(nullptr),
  511|  5.43M|          step(nullptr),
  512|  5.43M|          idFodder(id_fodder),
  513|  5.43M|          id(id)
  514|  5.43M|    {
  515|  5.43M|    }
_ZN7jsonnet8internal5IndexC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_bSE_SC_SE_SC_SE_SC_:
  520|  4.34M|        : AST(lr, AST_INDEX, open_fodder),
  521|  4.34M|          target(target),
  522|  4.34M|          dotFodder(dot_fodder),
  523|  4.34M|          isSlice(is_slice),
  524|  4.34M|          index(index),
  525|  4.34M|          endColonFodder(end_colon_fodder),
  526|  4.34M|          end(end),
  527|  4.34M|          stepColonFodder(step_colon_fodder),
  528|  4.34M|          step(step),
  529|  4.34M|          idFodder(id_fodder),
  530|  4.34M|          id(nullptr)
  531|  4.34M|    {
  532|  4.34M|    }
_ZN7jsonnet8internal5Local4BindC2ERKNSt3__16vectorINS0_13FodderElementENS3_9allocatorIS5_EEEEPKNS0_10IdentifierESA_PNS0_3ASTEbSA_RKNS4_INS0_8ArgParamENS6_ISG_EEEEbSA_SA_:
  551|  3.16M|            : varFodder(var_fodder),
  552|  3.16M|              var(var),
  553|  3.16M|              opFodder(op_fodder),
  554|  3.16M|              body(body),
  555|  3.16M|              functionSugar(function_sugar),
  556|  3.16M|              parenLeftFodder(paren_left_fodder),
  557|  3.16M|              params(params),
  558|  3.16M|              trailingComma(trailing_comma),
  559|  3.16M|              parenRightFodder(paren_right_fodder),
  560|  3.16M|              closeFodder(close_fodder)
  561|  3.16M|        {
  562|  3.16M|        }
_ZN7jsonnet8internal5LocalC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERKNS6_INS1_4BindENS8_ISD_EEEEPNS0_3ASTE:
  568|  3.47M|        : AST(lr, AST_LOCAL, open_fodder), binds(binds), body(body)
  569|  3.47M|    {
  570|  3.47M|    }
_ZN7jsonnet8internal14LiteralBooleanC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEb:
  577|   686k|        : AST(lr, AST_LITERAL_BOOLEAN, open_fodder), value(value)
  578|   686k|    {
  579|   686k|    }
_ZN7jsonnet8internal11LiteralNullC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEE:
  585|   397k|        : AST(lr, AST_LITERAL_NULL, open_fodder)
  586|   397k|    {
  587|   397k|    }
_ZN7jsonnet8internal13LiteralNumberC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERKNS5_12basic_stringIcNS5_11char_traitsIcEENS8_IcEEEE:
  595|  4.65M|        : AST(lr, AST_LITERAL_NUMBER, open_fodder),
  596|  4.65M|          value(strtod(str.c_str(), nullptr)),
  597|  4.65M|          originalString(str)
  598|  4.65M|    {
  599|  4.65M|    }
_ZN7jsonnet8internal13LiteralStringC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERKNS5_12basic_stringIDiNS5_11char_traitsIDiEENS8_IDiEEEENS1_9TokenKindERKNSD_IcNSE_IcEENS8_IcEEEESP_:
  612|  14.4M|        : AST(lr, AST_LITERAL_STRING, open_fodder),
  613|  14.4M|          value(value),
  614|  14.4M|          tokenKind(token_kind),
  615|  14.4M|          blockIndent(block_indent),
  616|  14.4M|          blockTermIndent(block_term_indent)
  617|  14.4M|    {
  618|  14.4M|    }
_ZN7jsonnet8internal11ObjectFieldC2ENS1_4KindERKNSt3__16vectorINS0_13FodderElementENS3_9allocatorIS5_EEEESA_SA_SA_NS1_4HideEbbPNS0_3ASTEPKNS0_10IdentifierERKNS0_13LocationRangeERKNS4_INS0_8ArgParamENS6_ISK_EEEEbSA_SD_SD_SA_:
  682|  1.92M|        : kind(kind),
  683|  1.92M|          fodder1(fodder1),
  684|  1.92M|          fodder2(fodder2),
  685|  1.92M|          fodderL(fodder_l),
  686|  1.92M|          fodderR(fodder_r),
  687|  1.92M|          hide(hide),
  688|  1.92M|          superSugar(super_sugar),
  689|  1.92M|          methodSugar(method_sugar),
  690|  1.92M|          expr1(expr1),
  691|  1.92M|          id(id),
  692|  1.92M|          idLocation(id_lr),
  693|  1.92M|          params(params),
  694|  1.92M|          trailingComma(trailing_comma),
  695|  1.92M|          opFodder(op_fodder),
  696|  1.92M|          expr2(expr2),
  697|  1.92M|          expr3(expr3),
  698|  1.92M|          commaFodder(comma_fodder)
  699|  1.92M|    {
  700|       |        // Enforce what is written in comments above.
  701|  1.92M|        assert(kind != ASSERT || (hide == VISIBLE && !superSugar && !methodSugar));
  ------------------
  |  Branch (701:9): [True: 43.3k, False: 0]
  |  Branch (701:9): [True: 43.3k, False: 0]
  |  Branch (701:9): [True: 43.3k, False: 0]
  |  Branch (701:9): [True: 1.88M, False: 43.3k]
  |  Branch (701:9): [True: 1.92M, False: 0]
  ------------------
  702|  1.92M|        assert(kind != LOCAL || (hide == VISIBLE && !superSugar));
  ------------------
  |  Branch (702:9): [True: 74.0k, False: 0]
  |  Branch (702:9): [True: 74.0k, False: 0]
  |  Branch (702:9): [True: 1.85M, False: 74.0k]
  |  Branch (702:9): [True: 1.92M, False: 0]
  ------------------
  703|  1.92M|        assert(kind != FIELD_ID || (id != nullptr && expr1 == nullptr));
  ------------------
  |  Branch (703:9): [True: 1.70M, False: 0]
  |  Branch (703:9): [True: 1.70M, False: 0]
  |  Branch (703:9): [True: 225k, False: 1.70M]
  |  Branch (703:9): [True: 1.92M, False: 0]
  ------------------
  704|  1.92M|        assert(kind == FIELD_ID || kind == LOCAL || id == nullptr);
  ------------------
  |  Branch (704:9): [True: 1.70M, False: 225k]
  |  Branch (704:9): [True: 74.0k, False: 151k]
  |  Branch (704:9): [True: 151k, False: 0]
  |  Branch (704:9): [True: 1.92M, False: 0]
  ------------------
  705|  1.92M|        assert(methodSugar || (params.size() == 0 && !trailingComma));
  ------------------
  |  Branch (705:9): [True: 980k, False: 0]
  |  Branch (705:9): [True: 980k, False: 0]
  |  Branch (705:9): [True: 946k, False: 980k]
  |  Branch (705:9): [True: 1.92M, False: 0]
  ------------------
  706|  1.92M|        assert(kind == ASSERT || expr3 == nullptr);
  ------------------
  |  Branch (706:9): [True: 43.3k, False: 1.88M]
  |  Branch (706:9): [True: 1.88M, False: 0]
  |  Branch (706:9): [True: 1.92M, False: 0]
  ------------------
  707|  1.92M|    }
_ZN7jsonnet8internal11ObjectField5LocalERKNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEES9_S9_S9_bPKNS0_10IdentifierERKNS3_INS0_8ArgParamENS5_ISD_EEEEbS9_PNS0_3ASTES9_:
  713|  53.0k|    {
  714|  53.0k|        return ObjectField(LOCAL,
  715|  53.0k|                           fodder1,
  716|  53.0k|                           fodder2,
  717|  53.0k|                           fodder_l,
  718|  53.0k|                           fodder_r,
  719|  53.0k|                           VISIBLE,
  720|  53.0k|                           false,
  721|  53.0k|                           method_sugar,
  722|  53.0k|                           nullptr,
  723|  53.0k|                           id,
  724|  53.0k|                           LocationRange(),
  725|  53.0k|                           params,
  726|  53.0k|                           trailing_comma,
  727|  53.0k|                           op_fodder,
  728|  53.0k|                           body,
  729|  53.0k|                           nullptr,
  730|  53.0k|                           comma_fodder);
  731|  53.0k|    }
_ZN7jsonnet8internal11ObjectField5LocalERKNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEES9_PKNS0_10IdentifierES9_PNS0_3ASTES9_:
  734|  20.9k|    {
  735|  20.9k|        return ObjectField(LOCAL,
  736|  20.9k|                           fodder1,
  737|  20.9k|                           fodder2,
  738|  20.9k|                           Fodder{},
  739|  20.9k|                           Fodder{},
  740|  20.9k|                           VISIBLE,
  741|  20.9k|                           false,
  742|  20.9k|                           false,
  743|  20.9k|                           nullptr,
  744|  20.9k|                           id,
  745|  20.9k|                           LocationRange(),
  746|  20.9k|                           ArgParams{},
  747|  20.9k|                           false,
  748|  20.9k|                           op_fodder,
  749|  20.9k|                           body,
  750|  20.9k|                           nullptr,
  751|  20.9k|                           comma_fodder);
  752|  20.9k|    }
_ZN7jsonnet8internal11ObjectField6AssertERKNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEEPNS0_3ASTES9_SB_S9_:
  779|  43.3k|    {
  780|  43.3k|        return ObjectField(ASSERT,
  781|  43.3k|                           fodder1,
  782|  43.3k|                           Fodder{},
  783|  43.3k|                           Fodder{},
  784|  43.3k|                           Fodder{},
  785|  43.3k|                           VISIBLE,
  786|  43.3k|                           false,
  787|  43.3k|                           false,
  788|  43.3k|                           nullptr,
  789|  43.3k|                           nullptr,
  790|  43.3k|                           LocationRange(),
  791|  43.3k|                           ArgParams{},
  792|  43.3k|                           false,
  793|  43.3k|                           op_fodder,
  794|  43.3k|                           body,
  795|  43.3k|                           msg,
  796|  43.3k|                           comma_fodder);
  797|  43.3k|    }
_ZN7jsonnet8internal6ObjectC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERKNS6_INS0_11ObjectFieldENS8_ISD_EEEEbSC_:
  812|   378k|        : AST(lr, AST_OBJECT, open_fodder),
  813|   378k|          fields(fields),
  814|   378k|          trailingComma(trailing_comma),
  815|   378k|          closeFodder(close_fodder)
  816|   378k|    {
  817|   378k|        assert(fields.size() > 0 || !trailing_comma);
  ------------------
  |  Branch (817:9): [True: 340k, False: 37.8k]
  |  Branch (817:9): [True: 37.8k, False: 0]
  |  Branch (817:9): [True: 378k, False: 0]
  ------------------
  818|   378k|        if (fields.size() > 0)
  ------------------
  |  Branch (818:13): [True: 340k, False: 37.8k]
  ------------------
  819|       |            assert(trailing_comma || fields[fields.size() - 1].commaFodder.size() == 0);
  ------------------
  |  Branch (819:13): [True: 105k, False: 234k]
  |  Branch (819:13): [True: 234k, False: 0]
  |  Branch (819:13): [True: 340k, False: 0]
  ------------------
  820|   378k|    }
_ZN7jsonnet8internal15DesugaredObject5FieldC2ENS0_11ObjectField4HideEPNS0_3ASTES6_:
  833|  2.04M|            : hide(hide), name(name), body(body)
  834|  2.04M|        {
  835|  2.04M|        }
_ZN7jsonnet8internal15DesugaredObjectC2ERKNS0_13LocationRangeERKNSt3__14listIPNS0_3ASTENS5_9allocatorIS8_EEEERKNS5_6vectorINS1_5FieldENS9_ISF_EEEE:
  841|   376k|        : AST(lr, AST_DESUGARED_OBJECT, Fodder{}), asserts(asserts), fields(fields)
  842|   376k|    {
  843|   376k|    }
_ZN7jsonnet8internal19ObjectComprehensionC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERKNS6_INS0_11ObjectFieldENS8_ISD_EEEEbRKNS6_INS0_17ComprehensionSpecENS8_ISI_EEEESC_:
  856|  35.3k|        : AST(lr, AST_OBJECT_COMPREHENSION, open_fodder),
  857|  35.3k|          fields(fields),
  858|  35.3k|          trailingComma(trailing_comma),
  859|  35.3k|          specs(specs),
  860|  35.3k|          closeFodder(close_fodder)
  861|  35.3k|    {
  862|  35.3k|    }
_ZN7jsonnet8internal25ObjectComprehensionSimpleC2ERKNS0_13LocationRangeEPNS0_3ASTES6_PKNS0_10IdentifierES6_:
  873|  31.6k|        : AST(lr, AST_OBJECT_COMPREHENSION_SIMPLE, Fodder{}),
  874|  31.6k|          field(field),
  875|  31.6k|          value(value),
  876|  31.6k|          id(id),
  877|  31.6k|          array(array)
  878|  31.6k|    {
  879|  31.6k|    }
_ZN7jsonnet8internal6ParensC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_:
  888|   378k|        : AST(lr, AST_PARENS, open_fodder), expr(expr), closeFodder(close_fodder)
  889|   378k|    {
  890|   378k|    }
_ZN7jsonnet8internal4SelfC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEE:
  895|  75.0k|    Self(const LocationRange &lr, const Fodder &open_fodder) : AST(lr, AST_SELF, open_fodder) {}
_ZN7jsonnet8internal10SuperIndexC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEESC_PNS0_3ASTESC_PKNS0_10IdentifierE:
  910|  12.8k|        : AST(lr, AST_SUPER_INDEX, open_fodder),
  911|  12.8k|          dotFodder(dot_fodder),
  912|  12.8k|          index(index),
  913|  12.8k|          idFodder(id_fodder),
  914|  12.8k|          id(id)
  915|  12.8k|    {
  916|  12.8k|    }
_ZN7jsonnet8internal7InSuperC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPNS0_3ASTESC_SC_:
  927|  10.0k|        : AST(lr, AST_IN_SUPER, open_fodder),
  928|  10.0k|          element(element),
  929|  10.0k|          inFodder(in_fodder),
  930|  10.0k|          superFodder(super_fodder)
  931|  10.0k|    {
  932|  10.0k|    }
_ZN7jsonnet8internal5UnaryC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEENS0_7UnaryOpEPNS0_3ASTE:
  956|   687k|        : AST(lr, AST_UNARY, open_fodder), op(op), expr(expr)
  957|   687k|    {
  958|   687k|    }
_ZN7jsonnet8internal3VarC2ERKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPKNS0_10IdentifierE:
  965|  29.7M|        : AST(lr, AST_VAR, open_fodder), id(id)
  966|  29.7M|    {
  967|  29.7M|    }
_ZN7jsonnet8internal9Allocator14makeIdentifierERKNSt3__112basic_stringIDiNS2_11char_traitsIDiEENS2_9allocatorIDiEEEE:
  998|  50.1M|    {
  999|  50.1M|        auto it = internedIdentifiers.find(name);
 1000|  50.1M|        if (it != internedIdentifiers.end()) {
  ------------------
  |  Branch (1000:13): [True: 48.1M, False: 2.01M]
  ------------------
 1001|  48.1M|            return it->second;
 1002|  48.1M|        }
 1003|  2.01M|        auto r = new Identifier(name);
 1004|  2.01M|        internedIdentifiers[name] = r;
 1005|  2.01M|        return r;
 1006|  50.1M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}
_ZN7jsonnet8internal9Allocator4makeINS0_6AssertEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_SF_SG_SF_EEEPT_DpOT0_:
  980|   310k|    {
  981|   310k|        auto r = new T(std::forward<Args>(args)...);
  982|   310k|        allocated.push_back(r);
  983|   310k|        return r;
  984|   310k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ErrorEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  980|   492k|    {
  981|   492k|        auto r = new T(std::forward<Args>(args)...);
  982|   492k|        allocated.push_back(r);
  983|   492k|        return r;
  984|   492k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_SF_SG_SF_EEEPT_DpOT0_:
  980|  2.43M|    {
  981|  2.43M|        auto r = new T(std::forward<Args>(args)...);
  982|  2.43M|        allocated.push_back(r);
  983|  2.43M|        return r;
  984|  2.43M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_SF_SA_DnEEEPT_DpOT0_:
  980|  30.3k|    {
  981|  30.3k|        auto r = new T(std::forward<Args>(args)...);
  982|  30.3k|        allocated.push_back(r);
  983|  30.3k|        return r;
  984|  30.3k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_8FunctionEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERSA_RNS6_INS0_8ArgParamENS8_ISE_EEEERbSD_RPNS0_3ASTEEEEPT_DpOT0_:
  980|   151k|    {
  981|   151k|        auto r = new T(std::forward<Args>(args)...);
  982|   151k|        allocated.push_back(r);
  983|   151k|        return r;
  984|   151k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6ImportEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_13LiteralStringEEEEPT_DpOT0_:
  980|    729|    {
  981|    729|        auto r = new T(std::forward<Args>(args)...);
  982|    729|        allocated.push_back(r);
  983|    729|        return r;
  984|    729|    }
_ZN7jsonnet8internal9Allocator4makeINS0_9ImportstrEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_13LiteralStringEEEEPT_DpOT0_:
  980|    570|    {
  981|    570|        auto r = new T(std::forward<Args>(args)...);
  982|    570|        allocated.push_back(r);
  983|    570|        return r;
  984|    570|    }
_ZN7jsonnet8internal9Allocator4makeINS0_9ImportbinEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_13LiteralStringEEEEPT_DpOT0_:
  980|    417|    {
  981|    417|        auto r = new T(std::forward<Args>(args)...);
  982|    417|        allocated.push_back(r);
  983|    417|        return r;
  984|    417|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5LocalEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS6_INS3_4BindENS8_ISD_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  980|  1.82M|    {
  981|  1.82M|        auto r = new T(std::forward<Args>(args)...);
  982|  1.82M|        allocated.push_back(r);
  983|  1.82M|        return r;
  984|  1.82M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5UnaryEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS0_7UnaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  980|   503k|    {
  981|   503k|        auto r = new T(std::forward<Args>(args)...);
  982|   503k|        allocated.push_back(r);
  983|   503k|        return r;
  984|   503k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6ObjectEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS6_INS0_11ObjectFieldENS8_ISD_EEEERbRSA_EEEPT_DpOT0_:
  980|   378k|    {
  981|   378k|        auto r = new T(std::forward<Args>(args)...);
  982|   378k|        allocated.push_back(r);
  983|   378k|        return r;
  984|   378k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_19ObjectComprehensionEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS6_INS0_11ObjectFieldENS8_ISD_EEEERbRNS6_INS0_17ComprehensionSpecENS8_ISI_EEEERSA_EEEPT_DpOT0_:
  980|  35.3k|    {
  981|  35.3k|        auto r = new T(std::forward<Args>(args)...);
  982|  35.3k|        allocated.push_back(r);
  983|  35.3k|        return r;
  984|  35.3k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralStringEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS6_12basic_stringIDiNS6_11char_traitsIDiEENS9_IDiEEEENS3_9TokenKindERA1_KcSL_EEEPT_DpOT0_:
  980|  69.3k|    {
  981|  69.3k|        auto r = new T(std::forward<Args>(args)...);
  982|  69.3k|        allocated.push_back(r);
  983|  69.3k|        return r;
  984|  69.3k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralStringEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS6_12basic_stringIDiNS6_11char_traitsIDiEENS9_IDiEEEENS3_9TokenKindERNSD_IcNSE_IcEENS9_IcEEEESM_EEEPT_DpOT0_:
  980|  1.17k|    {
  981|  1.17k|        auto r = new T(std::forward<Args>(args)...);
  982|  1.17k|        allocated.push_back(r);
  983|  1.17k|        return r;
  984|  1.17k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ArrayEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEENS6_INS3_7ElementENS8_ISC_EEEEbSB_EEEPT_DpOT0_:
  980|   172k|    {
  981|   172k|        auto r = new T(std::forward<Args>(args)...);
  982|   172k|        allocated.push_back(r);
  983|   172k|        return r;
  984|   172k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_18ArrayComprehensionEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTESB_RbRNS6_INS0_17ComprehensionSpecENS8_ISG_EEEESB_EEEPT_DpOT0_:
  980|   242k|    {
  981|   242k|        auto r = new T(std::forward<Args>(args)...);
  982|   242k|        allocated.push_back(r);
  983|   242k|        return r;
  984|   242k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ArrayEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS6_INS3_7ElementENS8_ISC_EEEERbSB_EEEPT_DpOT0_:
  980|   630k|    {
  981|   630k|        auto r = new T(std::forward<Args>(args)...);
  982|   630k|        allocated.push_back(r);
  983|   630k|        return r;
  984|   630k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6ParensEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTESB_EEEPT_DpOT0_:
  980|   378k|    {
  981|   378k|        auto r = new T(std::forward<Args>(args)...);
  982|   378k|        allocated.push_back(r);
  983|   378k|        return r;
  984|   378k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralNumberEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS5_12basic_stringIcNS5_11char_traitsIcEENS8_IcEEEEEEEPT_DpOT0_:
  980|  4.03M|    {
  981|  4.03M|        auto r = new T(std::forward<Args>(args)...);
  982|  4.03M|        allocated.push_back(r);
  983|  4.03M|        return r;
  984|  4.03M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralStringEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEENS5_12basic_stringIDiNS5_11char_traitsIDiEENS8_IDiEEEENS3_9TokenKindERA1_KcSK_EEEPT_DpOT0_:
  980|  3.60M|    {
  981|  3.60M|        auto r = new T(std::forward<Args>(args)...);
  982|  3.60M|        allocated.push_back(r);
  983|  3.60M|        return r;
  984|  3.60M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralStringEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEENS5_12basic_stringIDiNS5_11char_traitsIDiEENS8_IDiEEEENS3_9TokenKindERNSC_IcNSD_IcEENS8_IcEEEESL_EEEPT_DpOT0_:
  980|  3.62k|    {
  981|  3.62k|        auto r = new T(std::forward<Args>(args)...);
  982|  3.62k|        allocated.push_back(r);
  983|  3.62k|        return r;
  984|  3.62k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_14LiteralBooleanEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEbEEEPT_DpOT0_:
  980|   643k|    {
  981|   643k|        auto r = new T(std::forward<Args>(args)...);
  982|   643k|        allocated.push_back(r);
  983|   643k|        return r;
  984|   643k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11LiteralNullEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEEEEPT_DpOT0_:
  980|   138k|    {
  981|   138k|        auto r = new T(std::forward<Args>(args)...);
  982|   138k|        allocated.push_back(r);
  983|   138k|        return r;
  984|   138k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6DollarEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEEEEPT_DpOT0_:
  980|  26.0k|    {
  981|  26.0k|        auto r = new T(std::forward<Args>(args)...);
  982|  26.0k|        allocated.push_back(r);
  983|  26.0k|        return r;
  984|  26.0k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_3VarEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|  23.4M|    {
  981|  23.4M|        auto r = new T(std::forward<Args>(args)...);
  982|  23.4M|        allocated.push_back(r);
  983|  23.4M|        return r;
  984|  23.4M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_4SelfEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEEEEEPT_DpOT0_:
  980|  53.5k|    {
  981|  53.5k|        auto r = new T(std::forward<Args>(args)...);
  982|  53.5k|        allocated.push_back(r);
  983|  53.5k|        return r;
  984|  53.5k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_10SuperIndexEJNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEESB_RPNS0_3ASTESB_RPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|  2.94k|    {
  981|  2.94k|        auto r = new T(std::forward<Args>(args)...);
  982|  2.94k|        allocated.push_back(r);
  983|  2.94k|        return r;
  984|  2.94k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5IndexEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_RbSF_SG_SF_SG_SF_SG_EEEPT_DpOT0_:
  980|  1.31M|    {
  981|  1.31M|        auto r = new T(std::forward<Args>(args)...);
  982|  1.31M|        allocated.push_back(r);
  983|  1.31M|        return r;
  984|  1.31M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5IndexEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_SG_RPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|  5.43M|    {
  981|  5.43M|        auto r = new T(std::forward<Args>(args)...);
  982|  5.43M|        allocated.push_back(r);
  983|  5.43M|        return r;
  984|  5.43M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_RNS6_INS0_8ArgParamENS8_ISH_EEEERbSG_SG_SL_EEEPT_DpOT0_:
  980|  6.13M|    {
  981|  6.13M|        auto r = new T(std::forward<Args>(args)...);
  982|  6.13M|        allocated.push_back(r);
  983|  6.13M|        return r;
  984|  6.13M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_10ApplyBraceEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTESF_EEEPT_DpOT0_:
  980|  55.6k|    {
  981|  55.6k|        auto r = new T(std::forward<Args>(args)...);
  982|  55.6k|        allocated.push_back(r);
  983|  55.6k|        return r;
  984|  55.6k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_7InSuperEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_SG_EEEPT_DpOT0_:
  980|    115|    {
  981|    115|        auto r = new T(std::forward<Args>(args)...);
  982|    115|        allocated.push_back(r);
  983|    115|        return r;
  984|    115|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6BinaryEJNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPNS0_3ASTERSA_RNS0_8BinaryOpESF_EEEPT_DpOT0_:
  980|  6.56M|    {
  981|  6.56M|        auto r = new T(std::forward<Args>(args)...);
  982|  6.56M|        allocated.push_back(r);
  983|  6.56M|        return r;
  984|  6.56M|    }
pass.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
pass.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
pass.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}
_ZN7jsonnet8internal9Allocator5cloneINS0_5ApplyEEEPT_S5_:
  988|   560k|    {
  989|   560k|        auto r = new T(*ast);
  990|   560k|        allocated.push_back(r);
  991|   560k|        return r;
  992|   560k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_5ArrayEEEPT_S5_:
  988|   563k|    {
  989|   563k|        auto r = new T(*ast);
  990|   563k|        allocated.push_back(r);
  991|   563k|        return r;
  992|   563k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_6BinaryEEEPT_S5_:
  988|   971k|    {
  989|   971k|        auto r = new T(*ast);
  990|   971k|        allocated.push_back(r);
  991|   971k|        return r;
  992|   971k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_11ConditionalEEEPT_S5_:
  988|   502k|    {
  989|   502k|        auto r = new T(*ast);
  990|   502k|        allocated.push_back(r);
  991|   502k|        return r;
  992|   502k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_15DesugaredObjectEEEPT_S5_:
  988|   530k|    {
  989|   530k|        auto r = new T(*ast);
  990|   530k|        allocated.push_back(r);
  991|   530k|        return r;
  992|   530k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_5ErrorEEEPT_S5_:
  988|   127k|    {
  989|   127k|        auto r = new T(*ast);
  990|   127k|        allocated.push_back(r);
  991|   127k|        return r;
  992|   127k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_8FunctionEEEPT_S5_:
  988|  45.4k|    {
  989|  45.4k|        auto r = new T(*ast);
  990|  45.4k|        allocated.push_back(r);
  991|  45.4k|        return r;
  992|  45.4k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_6ImportEEEPT_S5_:
  988|    814|    {
  989|    814|        auto r = new T(*ast);
  990|    814|        allocated.push_back(r);
  991|    814|        return r;
  992|    814|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_9ImportstrEEEPT_S5_:
  988|  1.24k|    {
  989|  1.24k|        auto r = new T(*ast);
  990|  1.24k|        allocated.push_back(r);
  991|  1.24k|        return r;
  992|  1.24k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_9ImportbinEEEPT_S5_:
  988|  28.0k|    {
  989|  28.0k|        auto r = new T(*ast);
  990|  28.0k|        allocated.push_back(r);
  991|  28.0k|        return r;
  992|  28.0k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_5IndexEEEPT_S5_:
  988|   657k|    {
  989|   657k|        auto r = new T(*ast);
  990|   657k|        allocated.push_back(r);
  991|   657k|        return r;
  992|   657k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_7InSuperEEEPT_S5_:
  988|   140k|    {
  989|   140k|        auto r = new T(*ast);
  990|   140k|        allocated.push_back(r);
  991|   140k|        return r;
  992|   140k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_14LiteralBooleanEEEPT_S5_:
  988|  84.6k|    {
  989|  84.6k|        auto r = new T(*ast);
  990|  84.6k|        allocated.push_back(r);
  991|  84.6k|        return r;
  992|  84.6k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_11LiteralNullEEEPT_S5_:
  988|   101k|    {
  989|   101k|        auto r = new T(*ast);
  990|   101k|        allocated.push_back(r);
  991|   101k|        return r;
  992|   101k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_13LiteralNumberEEEPT_S5_:
  988|  1.02M|    {
  989|  1.02M|        auto r = new T(*ast);
  990|  1.02M|        allocated.push_back(r);
  991|  1.02M|        return r;
  992|  1.02M|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_13LiteralStringEEEPT_S5_:
  988|  1.58M|    {
  989|  1.58M|        auto r = new T(*ast);
  990|  1.58M|        allocated.push_back(r);
  991|  1.58M|        return r;
  992|  1.58M|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_5LocalEEEPT_S5_:
  988|   314k|    {
  989|   314k|        auto r = new T(*ast);
  990|   314k|        allocated.push_back(r);
  991|   314k|        return r;
  992|   314k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_25ObjectComprehensionSimpleEEEPT_S5_:
  988|  44.8k|    {
  989|  44.8k|        auto r = new T(*ast);
  990|  44.8k|        allocated.push_back(r);
  991|  44.8k|        return r;
  992|  44.8k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_4SelfEEEPT_S5_:
  988|  15.3k|    {
  989|  15.3k|        auto r = new T(*ast);
  990|  15.3k|        allocated.push_back(r);
  991|  15.3k|        return r;
  992|  15.3k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_10SuperIndexEEEPT_S5_:
  988|   148k|    {
  989|   148k|        auto r = new T(*ast);
  990|   148k|        allocated.push_back(r);
  991|   148k|        return r;
  992|   148k|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_5UnaryEEEPT_S5_:
  988|  2.86M|    {
  989|  2.86M|        auto r = new T(*ast);
  990|  2.86M|        allocated.push_back(r);
  991|  2.86M|        return r;
  992|  2.86M|    }
_ZN7jsonnet8internal9Allocator5cloneINS0_3VarEEEPT_S5_:
  988|  3.27M|    {
  989|  3.27M|        auto r = new T(*ast);
  990|  3.27M|        allocated.push_back(r);
  991|  3.27M|        return r;
  992|  3.27M|    }
static_analysis.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
static_analysis.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
static_analysis.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}
_ZN7jsonnet8internal9Allocator4makeINS0_3VarEJNS0_13LocationRangeENSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|  3.46k|    {
  981|  3.46k|        auto r = new T(std::forward<Args>(args)...);
  982|  3.46k|        allocated.push_back(r);
  983|  3.46k|        return r;
  984|  3.46k|    }
vm.cpp:_ZN7jsonnet8internalL10bop_stringENS0_8BinaryOpE:
  325|    176|{
  326|    176|    switch (bop) {
  327|      6|        case BOP_MULT: return "*";
  ------------------
  |  Branch (327:9): [True: 6, False: 170]
  ------------------
  328|      3|        case BOP_DIV: return "/";
  ------------------
  |  Branch (328:9): [True: 3, False: 173]
  ------------------
  329|      0|        case BOP_PERCENT: return "%";
  ------------------
  |  Branch (329:9): [True: 0, False: 176]
  ------------------
  330|       |
  331|     44|        case BOP_PLUS: return "+";
  ------------------
  |  Branch (331:9): [True: 44, False: 132]
  ------------------
  332|     22|        case BOP_MINUS: return "-";
  ------------------
  |  Branch (332:9): [True: 22, False: 154]
  ------------------
  333|       |
  334|      4|        case BOP_SHIFT_L: return "<<";
  ------------------
  |  Branch (334:9): [True: 4, False: 172]
  ------------------
  335|      1|        case BOP_SHIFT_R: return ">>";
  ------------------
  |  Branch (335:9): [True: 1, False: 175]
  ------------------
  336|       |
  337|     19|        case BOP_GREATER: return ">";
  ------------------
  |  Branch (337:9): [True: 19, False: 157]
  ------------------
  338|     15|        case BOP_GREATER_EQ: return ">=";
  ------------------
  |  Branch (338:9): [True: 15, False: 161]
  ------------------
  339|     19|        case BOP_LESS: return "<";
  ------------------
  |  Branch (339:9): [True: 19, False: 157]
  ------------------
  340|     15|        case BOP_LESS_EQ: return "<=";
  ------------------
  |  Branch (340:9): [True: 15, False: 161]
  ------------------
  341|      0|        case BOP_IN: return "in";
  ------------------
  |  Branch (341:9): [True: 0, False: 176]
  ------------------
  342|       |
  343|      0|        case BOP_MANIFEST_EQUAL: return "==";
  ------------------
  |  Branch (343:9): [True: 0, False: 176]
  ------------------
  344|      0|        case BOP_MANIFEST_UNEQUAL: return "!=";
  ------------------
  |  Branch (344:9): [True: 0, False: 176]
  ------------------
  345|       |
  346|     12|        case BOP_BITWISE_AND: return "&";
  ------------------
  |  Branch (346:9): [True: 12, False: 164]
  ------------------
  347|      5|        case BOP_BITWISE_XOR: return "^";
  ------------------
  |  Branch (347:9): [True: 5, False: 171]
  ------------------
  348|      5|        case BOP_BITWISE_OR: return "|";
  ------------------
  |  Branch (348:9): [True: 5, False: 171]
  ------------------
  349|       |
  350|      1|        case BOP_AND: return "&&";
  ------------------
  |  Branch (350:9): [True: 1, False: 175]
  ------------------
  351|      5|        case BOP_OR: return "||";
  ------------------
  |  Branch (351:9): [True: 5, False: 171]
  ------------------
  352|       |
  353|      0|        default:
  ------------------
  |  Branch (353:9): [True: 0, False: 176]
  ------------------
  354|      0|            std::cerr << "INTERNAL ERROR: Unrecognised binary operator: " << bop << std::endl;
  355|      0|            std::abort();
  356|    176|    }
  357|    176|}
vm.cpp:_ZN7jsonnet8internalL10uop_stringENS0_7UnaryOpE:
  938|     99|{
  939|     99|    switch (uop) {
  940|     73|        case UOP_PLUS: return "+";
  ------------------
  |  Branch (940:9): [True: 73, False: 26]
  ------------------
  941|      5|        case UOP_MINUS: return "-";
  ------------------
  |  Branch (941:9): [True: 5, False: 94]
  ------------------
  942|      0|        case UOP_BITWISE_NOT: return "~";
  ------------------
  |  Branch (942:9): [True: 0, False: 99]
  ------------------
  943|     21|        case UOP_NOT: return "!";
  ------------------
  |  Branch (943:9): [True: 21, False: 78]
  ------------------
  944|       |
  945|      0|        default:
  ------------------
  |  Branch (945:9): [True: 0, False: 99]
  ------------------
  946|      0|            std::cerr << "INTERNAL ERROR: Unrecognised unary operator: " << uop << std::endl;
  947|      0|            std::abort();
  948|     99|    }
  949|     99|}
desugarer.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
desugarer.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
desugarer.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}
_ZN7jsonnet8internal9Allocator4makeINS0_6BinaryEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTERKSB_NS0_8BinaryOpESF_EEEPT_DpOT0_:
  980|  54.9k|    {
  981|  54.9k|        auto r = new T(std::forward<Args>(args)...);
  982|  54.9k|        allocated.push_back(r);
  983|  54.9k|        return r;
  984|  54.9k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralNumberEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERA4_KcEEEPT_DpOT0_:
  980|   578k|    {
  981|   578k|        auto r = new T(std::forward<Args>(args)...);
  982|   578k|        allocated.push_back(r);
  983|   578k|        return r;
  984|   578k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_3VarESD_NS7_INS0_8ArgParamENS9_ISG_EEEEbSD_SD_bEEEPT_DpOT0_:
  980|   273k|    {
  981|   273k|        auto r = new T(std::forward<Args>(args)...);
  982|   273k|        allocated.push_back(r);
  983|   273k|        return r;
  984|   273k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6BinaryEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS0_8BinaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  980|   369k|    {
  981|   369k|        auto r = new T(std::forward<Args>(args)...);
  982|   369k|        allocated.push_back(r);
  983|   369k|        return r;
  984|   369k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6BinaryEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS0_8BinaryOpEPNS0_5ArrayEEEEPT_DpOT0_:
  980|   273k|    {
  981|   273k|        auto r = new T(std::forward<Args>(args)...);
  982|   273k|        allocated.push_back(r);
  983|   273k|        return r;
  984|   273k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ArrayEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_7ElementENS9_ISE_EEEEbSD_EEEPT_DpOT0_:
  980|   273k|    {
  981|   273k|        auto r = new T(std::forward<Args>(args)...);
  982|   273k|        allocated.push_back(r);
  983|   273k|        return r;
  984|   273k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS8_INS0_8ArgParamENSA_ISH_EEEEbSE_SE_bEEEPT_DpOT0_:
  980|   395k|    {
  981|   395k|        auto r = new T(std::forward<Args>(args)...);
  982|   395k|        allocated.push_back(r);
  983|   395k|        return r;
  984|   395k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERKPNS0_3ASTESD_RSF_SD_SI_EEEPT_DpOT0_:
  980|  69.6k|    {
  981|  69.6k|        auto r = new T(std::forward<Args>(args)...);
  982|  69.6k|        allocated.push_back(r);
  983|  69.6k|        return r;
  984|  69.6k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5LocalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_4BindENS9_ISE_EEEEPNS0_11ConditionalEEEEPT_DpOT0_:
  980|   299k|    {
  981|   299k|        auto r = new T(std::forward<Args>(args)...);
  982|   299k|        allocated.push_back(r);
  983|   299k|        return r;
  984|   299k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_8FunctionEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEESD_NS7_INS0_8ArgParamENS9_ISE_EEEEbSD_PNS0_11ConditionalEEEEPT_DpOT0_:
  980|   299k|    {
  981|   299k|        auto r = new T(std::forward<Args>(args)...);
  982|   299k|        allocated.push_back(r);
  983|   299k|        return r;
  984|   299k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_6BinaryESD_RPNS0_3ASTESD_PNS0_5LocalEEEEPT_DpOT0_:
  980|   299k|    {
  981|   299k|        auto r = new T(std::forward<Args>(args)...);
  982|   299k|        allocated.push_back(r);
  983|   299k|        return r;
  984|   299k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6BinaryEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS0_8BinaryOpEPNS0_5ApplyEEEEPT_DpOT0_:
  980|   299k|    {
  981|   299k|        auto r = new T(std::forward<Args>(args)...);
  982|   299k|        allocated.push_back(r);
  983|   299k|        return r;
  984|   299k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5LocalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_4BindENS9_ISE_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  980|   308k|    {
  981|   308k|        auto r = new T(std::forward<Args>(args)...);
  982|   308k|        allocated.push_back(r);
  983|   308k|        return r;
  984|   308k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bSG_SE_DnSE_DnSE_EEEPT_DpOT0_:
  980|   299k|    {
  981|   299k|        auto r = new T(std::forward<Args>(args)...);
  982|   299k|        allocated.push_back(r);
  983|   299k|        return r;
  984|   299k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_5ApplyESD_SF_SD_PNS0_5ErrorEEEEPT_DpOT0_:
  980|   299k|    {
  981|   299k|        auto r = new T(std::forward<Args>(args)...);
  982|   299k|        allocated.push_back(r);
  983|   299k|        return r;
  984|   299k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ArrayEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEENS8_INS3_7ElementENSA_ISF_EEEEbSE_EEEPT_DpOT0_:
  980|   273k|    {
  981|   273k|        auto r = new T(std::forward<Args>(args)...);
  982|   273k|        allocated.push_back(r);
  983|   273k|        return r;
  984|   273k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralStringEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERKNS7_12basic_stringIDiNS7_11char_traitsIDiEENSA_IDiEEEENS3_9TokenKindERA1_KcSP_EEEPT_DpOT0_:
  980|  10.7M|    {
  981|  10.7M|        auto r = new T(std::forward<Args>(args)...);
  982|  10.7M|        allocated.push_back(r);
  983|  10.7M|        return r;
  984|  10.7M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ErrorEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  980|   653k|    {
  981|   653k|        auto r = new T(std::forward<Args>(args)...);
  982|   653k|        allocated.push_back(r);
  983|   653k|        return r;
  984|   653k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTERKSB_SF_SH_SF_EEEPT_DpOT0_:
  980|   310k|    {
  981|   310k|        auto r = new T(std::forward<Args>(args)...);
  982|   310k|        allocated.push_back(r);
  983|   310k|        return r;
  984|   310k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bPNS0_13LiteralStringESE_DnSE_DnSE_EEEPT_DpOT0_:
  980|  2.66M|    {
  981|  2.66M|        auto r = new T(std::forward<Args>(args)...);
  982|  2.66M|        allocated.push_back(r);
  983|  2.66M|        return r;
  984|  2.66M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTERKSB_RNS7_INS0_8ArgParamENS9_ISI_EEEEbSH_SH_bEEEPT_DpOT0_:
  980|   239k|    {
  981|   239k|        auto r = new T(std::forward<Args>(args)...);
  982|   239k|        allocated.push_back(r);
  983|   239k|        return r;
  984|   239k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_5IndexESE_NS8_INS0_8ArgParamENSA_ISH_EEEEbSE_SE_bEEEPT_DpOT0_:
  980|  1.67M|    {
  981|  1.67M|        auto r = new T(std::forward<Args>(args)...);
  982|  1.67M|        allocated.push_back(r);
  983|  1.67M|        return r;
  984|  1.67M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5UnaryEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS0_7UnaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  980|   183k|    {
  981|   183k|        auto r = new T(std::forward<Args>(args)...);
  982|   183k|        allocated.push_back(r);
  983|   183k|        return r;
  984|   183k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11LiteralNullEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEEEEPT_DpOT0_:
  980|   258k|    {
  981|   258k|        auto r = new T(std::forward<Args>(args)...);
  982|   258k|        allocated.push_back(r);
  983|   258k|        return r;
  984|   258k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_3VarEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|  6.00M|    {
  981|  6.00M|        auto r = new T(std::forward<Args>(args)...);
  982|  6.00M|        allocated.push_back(r);
  983|  6.00M|        return r;
  984|  6.00M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_5IndexESD_NS7_INS0_8ArgParamENS9_ISG_EEEEbSD_SD_bEEEPT_DpOT0_:
  980|   743k|    {
  981|   743k|        auto r = new T(std::forward<Args>(args)...);
  982|   743k|        allocated.push_back(r);
  983|   743k|        return r;
  984|   743k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_8FunctionEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEESC_RNS7_INS0_8ArgParamENS9_ISD_EEEEbSC_RPNS0_3ASTEEEEPT_DpOT0_:
  980|   559k|    {
  981|   559k|        auto r = new T(std::forward<Args>(args)...);
  982|   559k|        allocated.push_back(r);
  983|   559k|        return r;
  984|   559k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_4SelfEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEEEEPT_DpOT0_:
  980|  21.5k|    {
  981|  21.5k|        auto r = new T(std::forward<Args>(args)...);
  982|  21.5k|        allocated.push_back(r);
  983|  21.5k|        return r;
  984|  21.5k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTESD_PNS0_14LiteralBooleanESD_PNS0_5ErrorEEEEPT_DpOT0_:
  980|  42.9k|    {
  981|  42.9k|        auto r = new T(std::forward<Args>(args)...);
  982|  42.9k|        allocated.push_back(r);
  983|  42.9k|        return r;
  984|  42.9k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_14LiteralBooleanEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEbEEEPT_DpOT0_:
  980|  42.9k|    {
  981|  42.9k|        auto r = new T(std::forward<Args>(args)...);
  982|  42.9k|        allocated.push_back(r);
  983|  42.9k|        return r;
  984|  42.9k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_8FunctionEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERSB_RNS7_INS0_8ArgParamENS9_ISF_EEEERbSE_RPNS0_3ASTEEEEPT_DpOT0_:
  980|   946k|    {
  981|   946k|        auto r = new T(std::forward<Args>(args)...);
  982|   946k|        allocated.push_back(r);
  983|   946k|        return r;
  984|   946k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_3VarEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|   226k|    {
  981|   226k|        auto r = new T(std::forward<Args>(args)...);
  982|   226k|        allocated.push_back(r);
  983|   226k|        return r;
  984|   226k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_7InSuperESD_PNS0_6BinaryESD_PNS0_3ASTEEEEPT_DpOT0_:
  980|  9.91k|    {
  981|  9.91k|        auto r = new T(std::forward<Args>(args)...);
  982|  9.91k|        allocated.push_back(r);
  983|  9.91k|        return r;
  984|  9.91k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_7InSuperEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTESD_SD_EEEPT_DpOT0_:
  980|  9.91k|    {
  981|  9.91k|        auto r = new T(std::forward<Args>(args)...);
  982|  9.91k|        allocated.push_back(r);
  983|  9.91k|        return r;
  984|  9.91k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_6BinaryEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_10SuperIndexESD_NS0_8BinaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  980|  9.91k|    {
  981|  9.91k|        auto r = new T(std::forward<Args>(args)...);
  982|  9.91k|        allocated.push_back(r);
  983|  9.91k|        return r;
  984|  9.91k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_10SuperIndexEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEESD_PNS0_3ASTESD_DnEEEPT_DpOT0_:
  980|  9.91k|    {
  981|  9.91k|        auto r = new T(std::forward<Args>(args)...);
  982|  9.91k|        allocated.push_back(r);
  983|  9.91k|        return r;
  984|  9.91k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_15DesugaredObjectEJRNS0_13LocationRangeERNSt3__14listIPNS0_3ASTENS6_9allocatorIS9_EEEERNS6_6vectorINS3_5FieldENSA_ISF_EEEEEEEPT_DpOT0_:
  980|   376k|    {
  981|   376k|        auto r = new T(std::forward<Args>(args)...);
  982|   376k|        allocated.push_back(r);
  983|   376k|        return r;
  984|   376k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5LocalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERNS7_INS3_4BindENS9_ISE_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  980|  1.03M|    {
  981|  1.03M|        auto r = new T(std::forward<Args>(args)...);
  982|  1.03M|        allocated.push_back(r);
  983|  1.03M|        return r;
  984|  1.03M|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bPNS0_13LiteralNumberESE_DnSE_DnSE_EEEPT_DpOT0_:
  980|  35.2k|    {
  981|  35.2k|        auto r = new T(std::forward<Args>(args)...);
  982|  35.2k|        allocated.push_back(r);
  983|  35.2k|        return r;
  984|  35.2k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_13LiteralNumberEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEENS7_12basic_stringIcNS7_11char_traitsIcEENSA_IcEEEEEEEPT_DpOT0_:
  980|  35.2k|    {
  981|  35.2k|        auto r = new T(std::forward<Args>(args)...);
  982|  35.2k|        allocated.push_back(r);
  983|  35.2k|        return r;
  984|  35.2k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_18ArrayComprehensionEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_5ArrayESD_bRNS7_INS0_17ComprehensionSpecENS9_ISG_EEEESD_EEEPT_DpOT0_:
  980|  31.6k|    {
  981|  31.6k|        auto r = new T(std::forward<Args>(args)...);
  982|  31.6k|        allocated.push_back(r);
  983|  31.6k|        return r;
  984|  31.6k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ArrayEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERNS7_INS3_7ElementENS9_ISE_EEEEbSD_EEEPT_DpOT0_:
  980|  31.6k|    {
  981|  31.6k|        auto r = new T(std::forward<Args>(args)...);
  982|  31.6k|        allocated.push_back(r);
  983|  31.6k|        return r;
  984|  31.6k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_25ObjectComprehensionSimpleEJRNS0_13LocationRangeEPNS0_5IndexEPNS0_5LocalERPKNS0_10IdentifierERPNS0_3ASTEEEEPT_DpOT0_:
  980|  31.6k|    {
  981|  31.6k|        auto r = new T(std::forward<Args>(args)...);
  982|  31.6k|        allocated.push_back(r);
  983|  31.6k|        return r;
  984|  31.6k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bRPNS0_3ASTESE_DnSE_DnSE_EEEPT_DpOT0_:
  980|  31.6k|    {
  981|  31.6k|        auto r = new T(std::forward<Args>(args)...);
  982|  31.6k|        allocated.push_back(r);
  983|  31.6k|        return r;
  984|  31.6k|    }
_ZN7jsonnet8internal9Allocator15makeBuiltinBodyIZNS0_9Desugarer9stdlibASTENSt3__112basic_stringIcNS4_11char_traitsIcEENS4_9allocatorIcEEEEEUlvE_EEPKNS0_19BuiltinFunctionBodyERKNS0_13LocationRangeERKSA_OT_:
 1009|   317k|    {
 1010|   317k|        auto it = internedBuiltins.find(name);
 1011|   317k|        if (it != internedBuiltins.end()) {
  ------------------
  |  Branch (1011:13): [True: 145k, False: 172k]
  ------------------
 1012|   145k|            return it->second;
 1013|   145k|        }
 1014|   172k|        const Identifiers params = make_params();
 1015|   172k|        auto r = new BuiltinFunctionBody(loc, name, params);
 1016|   172k|        internedBuiltins[name] = r;
 1017|   172k|        return r;
 1018|   317k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_15BuiltinFunctionEJRKNS0_13LocationRangeERPKNS0_19BuiltinFunctionBodyEEEEPT_DpOT0_:
  980|   317k|    {
  981|   317k|        auto r = new T(std::forward<Args>(args)...);
  982|   317k|        allocated.push_back(r);
  983|   317k|        return r;
  984|   317k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5LocalEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_4BindENS9_ISD_EEEEPNS0_11ConditionalEEEEPT_DpOT0_:
  980|  4.10k|    {
  981|  4.10k|        auto r = new T(std::forward<Args>(args)...);
  982|  4.10k|        allocated.push_back(r);
  983|  4.10k|        return r;
  984|  4.10k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_11ConditionalEJRKNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_5ApplyERKSC_SF_SD_PNS0_3VarEEEEPT_DpOT0_:
  980|  4.10k|    {
  981|  4.10k|        auto r = new T(std::forward<Args>(args)...);
  982|  4.10k|        allocated.push_back(r);
  983|  4.10k|        return r;
  984|  4.10k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_5ApplyEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_3VarESD_RNS7_INS0_8ArgParamENS9_ISG_EEEEbSD_SD_bEEEPT_DpOT0_:
  980|  4.10k|    {
  981|  4.10k|        auto r = new T(std::forward<Args>(args)...);
  982|  4.10k|        allocated.push_back(r);
  983|  4.10k|        return r;
  984|  4.10k|    }
_ZN7jsonnet8internal9Allocator4makeINS0_3VarEJRKNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERPKNS0_10IdentifierEEEEPT_DpOT0_:
  980|  8.20k|    {
  981|  8.20k|        auto r = new T(std::forward<Args>(args)...);
  982|  8.20k|        allocated.push_back(r);
  983|  8.20k|        return r;
  984|  8.20k|    }
formatter.cpp:_ZN7jsonnet8internal12_GLOBAL__N_120build_precedence_mapEv:
 1047|      2|{
 1048|      2|    std::map<BinaryOp, int> r;
 1049|       |
 1050|      2|    r[BOP_MULT] = 5;
 1051|      2|    r[BOP_DIV] = 5;
 1052|      2|    r[BOP_PERCENT] = 5;
 1053|       |
 1054|      2|    r[BOP_PLUS] = 6;
 1055|      2|    r[BOP_MINUS] = 6;
 1056|       |
 1057|      2|    r[BOP_SHIFT_L] = 7;
 1058|      2|    r[BOP_SHIFT_R] = 7;
 1059|       |
 1060|      2|    r[BOP_GREATER] = 8;
 1061|      2|    r[BOP_GREATER_EQ] = 8;
 1062|      2|    r[BOP_LESS] = 8;
 1063|      2|    r[BOP_LESS_EQ] = 8;
 1064|      2|    r[BOP_IN] = 8;
 1065|       |
 1066|      2|    r[BOP_MANIFEST_EQUAL] = 9;
 1067|      2|    r[BOP_MANIFEST_UNEQUAL] = 9;
 1068|       |
 1069|      2|    r[BOP_BITWISE_AND] = 10;
 1070|       |
 1071|      2|    r[BOP_BITWISE_XOR] = 11;
 1072|       |
 1073|      2|    r[BOP_BITWISE_OR] = 12;
 1074|       |
 1075|      2|    r[BOP_AND] = 13;
 1076|       |
 1077|      2|    r[BOP_OR] = 14;
 1078|       |
 1079|      2|    return r;
 1080|      2|}
formatter.cpp:_ZN7jsonnet8internal12_GLOBAL__N_115build_unary_mapEv:
 1083|      2|{
 1084|      2|    std::map<std::string, UnaryOp> r;
 1085|      2|    r["!"] = UOP_NOT;
 1086|      2|    r["~"] = UOP_BITWISE_NOT;
 1087|      2|    r["+"] = UOP_PLUS;
 1088|      2|    r["-"] = UOP_MINUS;
 1089|      2|    return r;
 1090|      2|}
formatter.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116build_binary_mapEv:
 1093|      2|{
 1094|      2|    std::map<std::string, BinaryOp> r;
 1095|       |
 1096|      2|    r["*"] = BOP_MULT;
 1097|      2|    r["/"] = BOP_DIV;
 1098|      2|    r["%"] = BOP_PERCENT;
 1099|       |
 1100|      2|    r["+"] = BOP_PLUS;
 1101|      2|    r["-"] = BOP_MINUS;
 1102|       |
 1103|      2|    r["<<"] = BOP_SHIFT_L;
 1104|      2|    r[">>"] = BOP_SHIFT_R;
 1105|       |
 1106|      2|    r[">"] = BOP_GREATER;
 1107|      2|    r[">="] = BOP_GREATER_EQ;
 1108|      2|    r["<"] = BOP_LESS;
 1109|      2|    r["<="] = BOP_LESS_EQ;
 1110|      2|    r["in"] = BOP_IN;
 1111|       |
 1112|      2|    r["=="] = BOP_MANIFEST_EQUAL;
 1113|      2|    r["!="] = BOP_MANIFEST_UNEQUAL;
 1114|       |
 1115|      2|    r["&"] = BOP_BITWISE_AND;
 1116|      2|    r["^"] = BOP_BITWISE_XOR;
 1117|      2|    r["|"] = BOP_BITWISE_OR;
 1118|       |
 1119|      2|    r["&&"] = BOP_AND;
 1120|      2|    r["||"] = BOP_OR;
 1121|      2|    return r;
 1122|      2|}

_ZN7jsonnet8internal20jsonnet_builtin_declEm:
   41|   317k|{
   42|   317k|    switch (builtin) {
   43|  7.56k|        case 0: return {U"makeArray", {U"sz", U"func"}};
  ------------------
  |  Branch (43:9): [True: 7.56k, False: 309k]
  ------------------
   44|  7.56k|        case 1: return {U"pow", {U"x", U"n"}};
  ------------------
  |  Branch (44:9): [True: 7.56k, False: 309k]
  ------------------
   45|  7.56k|        case 2: return {U"floor", {U"x"}};
  ------------------
  |  Branch (45:9): [True: 7.56k, False: 309k]
  ------------------
   46|  7.56k|        case 3: return {U"ceil", {U"x"}};
  ------------------
  |  Branch (46:9): [True: 7.56k, False: 309k]
  ------------------
   47|  7.56k|        case 4: return {U"sqrt", {U"x"}};
  ------------------
  |  Branch (47:9): [True: 7.56k, False: 309k]
  ------------------
   48|  7.56k|        case 5: return {U"sin", {U"x"}};
  ------------------
  |  Branch (48:9): [True: 7.56k, False: 309k]
  ------------------
   49|  7.56k|        case 6: return {U"cos", {U"x"}};
  ------------------
  |  Branch (49:9): [True: 7.56k, False: 309k]
  ------------------
   50|  7.56k|        case 7: return {U"tan", {U"x"}};
  ------------------
  |  Branch (50:9): [True: 7.56k, False: 309k]
  ------------------
   51|  7.56k|        case 8: return {U"asin", {U"x"}};
  ------------------
  |  Branch (51:9): [True: 7.56k, False: 309k]
  ------------------
   52|  7.56k|        case 9: return {U"acos", {U"x"}};
  ------------------
  |  Branch (52:9): [True: 7.56k, False: 309k]
  ------------------
   53|  7.56k|        case 10: return {U"atan", {U"x"}};
  ------------------
  |  Branch (53:9): [True: 7.56k, False: 309k]
  ------------------
   54|  7.56k|        case 11: return {U"type", {U"x"}};
  ------------------
  |  Branch (54:9): [True: 7.56k, False: 309k]
  ------------------
   55|  7.56k|        case 12: return {U"filter", {U"func", U"arr"}};
  ------------------
  |  Branch (55:9): [True: 7.56k, False: 309k]
  ------------------
   56|  7.56k|        case 13: return {U"objectHasEx", {U"obj", U"f", U"inc_hidden"}};
  ------------------
  |  Branch (56:9): [True: 7.56k, False: 309k]
  ------------------
   57|  7.56k|        case 14: return {U"length", {U"x"}};
  ------------------
  |  Branch (57:9): [True: 7.56k, False: 309k]
  ------------------
   58|  7.56k|        case 15: return {U"objectFieldsEx", {U"obj", U"inc_hidden"}};
  ------------------
  |  Branch (58:9): [True: 7.56k, False: 309k]
  ------------------
   59|  7.56k|        case 16: return {U"codepoint", {U"str"}};
  ------------------
  |  Branch (59:9): [True: 7.56k, False: 309k]
  ------------------
   60|  7.56k|        case 17: return {U"char", {U"n"}};
  ------------------
  |  Branch (60:9): [True: 7.56k, False: 309k]
  ------------------
   61|  7.56k|        case 18: return {U"log", {U"n"}};
  ------------------
  |  Branch (61:9): [True: 7.56k, False: 309k]
  ------------------
   62|  7.56k|        case 19: return {U"exp", {U"n"}};
  ------------------
  |  Branch (62:9): [True: 7.56k, False: 309k]
  ------------------
   63|  7.56k|        case 20: return {U"mantissa", {U"n"}};
  ------------------
  |  Branch (63:9): [True: 7.56k, False: 309k]
  ------------------
   64|  7.56k|        case 21: return {U"exponent", {U"n"}};
  ------------------
  |  Branch (64:9): [True: 7.56k, False: 309k]
  ------------------
   65|  7.56k|        case 22: return {U"modulo", {U"a", U"b"}};
  ------------------
  |  Branch (65:9): [True: 7.56k, False: 309k]
  ------------------
   66|  7.56k|        case 23: return {U"extVar", {U"x"}};
  ------------------
  |  Branch (66:9): [True: 7.56k, False: 309k]
  ------------------
   67|  7.56k|        case 24: return {U"primitiveEquals", {U"a", U"b"}};
  ------------------
  |  Branch (67:9): [True: 7.56k, False: 309k]
  ------------------
   68|  7.56k|        case 25: return {U"native", {U"name"}};
  ------------------
  |  Branch (68:9): [True: 7.56k, False: 309k]
  ------------------
   69|  7.56k|        case 26: return {U"md5", {U"str"}};
  ------------------
  |  Branch (69:9): [True: 7.56k, False: 309k]
  ------------------
   70|  7.56k|        case 27: return {U"trace", {U"str", U"rest"}};
  ------------------
  |  Branch (70:9): [True: 7.56k, False: 309k]
  ------------------
   71|  7.56k|        case 28: return {U"splitLimit", {U"str", U"c", U"maxsplits"}};
  ------------------
  |  Branch (71:9): [True: 7.56k, False: 309k]
  ------------------
   72|  7.56k|        case 29: return {U"substr", {U"str", U"from", U"len"}};
  ------------------
  |  Branch (72:9): [True: 7.56k, False: 309k]
  ------------------
   73|  7.56k|        case 30: return {U"range", {U"from", U"to"}};
  ------------------
  |  Branch (73:9): [True: 7.56k, False: 309k]
  ------------------
   74|  7.56k|        case 31: return {U"strReplace", {U"str", U"from", U"to"}};
  ------------------
  |  Branch (74:9): [True: 7.56k, False: 309k]
  ------------------
   75|  7.56k|        case 32: return {U"asciiLower", {U"str"}};
  ------------------
  |  Branch (75:9): [True: 7.56k, False: 309k]
  ------------------
   76|  7.56k|        case 33: return {U"asciiUpper", {U"str"}};
  ------------------
  |  Branch (76:9): [True: 7.56k, False: 309k]
  ------------------
   77|  7.56k|        case 34: return {U"join", {U"sep", U"arr"}};
  ------------------
  |  Branch (77:9): [True: 7.56k, False: 309k]
  ------------------
   78|  7.56k|        case 35: return {U"parseJson", {U"str"}};
  ------------------
  |  Branch (78:9): [True: 7.56k, False: 309k]
  ------------------
   79|  7.56k|        case 36: return {U"parseYaml", {U"str"}};
  ------------------
  |  Branch (79:9): [True: 7.56k, False: 309k]
  ------------------
   80|  7.56k|        case 37: return {U"encodeUTF8", {U"str"}};
  ------------------
  |  Branch (80:9): [True: 7.56k, False: 309k]
  ------------------
   81|  7.56k|        case 38: return {U"decodeUTF8", {U"arr"}};
  ------------------
  |  Branch (81:9): [True: 7.56k, False: 309k]
  ------------------
   82|  7.56k|        case 39: return {U"atan2", {U"y", U"x"}};
  ------------------
  |  Branch (82:9): [True: 7.56k, False: 309k]
  ------------------
   83|  7.56k|        case 40: return {U"hypot", {U"a", U"b"}};
  ------------------
  |  Branch (83:9): [True: 7.56k, False: 309k]
  ------------------
   84|  7.56k|        case 41: return {U"objectRemoveKey", {U"obj", U"key"}};
  ------------------
  |  Branch (84:9): [True: 7.56k, False: 309k]
  ------------------
   85|      0|        default:
  ------------------
  |  Branch (85:9): [True: 0, False: 317k]
  ------------------
   86|      0|            std::cerr << "INTERNAL ERROR: Unrecognized builtin function: " << builtin << std::endl;
   87|      0|            std::abort();
   88|   317k|    }
   89|       |    // Quiet, compiler.
   90|      0|    return BuiltinDecl();
   91|   317k|}
_ZN7jsonnet8internal13makeStdlibASTEPNS0_9AllocatorENSt3__112basic_stringIcNS3_11char_traitsIcEENS3_9allocatorIcEEEE:
 1022|  3.46k|DesugaredObject *makeStdlibAST(Allocator *alloc, std::string filename) {
 1023|  3.46k|    Desugarer desugarer(alloc, true);
 1024|  3.46k|    return desugarer.stdlibAST(filename);
 1025|  3.46k|}
_ZN7jsonnet8internal15jsonnet_desugarEPNS0_9AllocatorERPNS0_3ASTEPNSt3__13mapINS6_12basic_stringIcNS6_11char_traitsIcEENS6_9allocatorIcEEEENS0_5VmExtENS6_4lessISD_EENSB_INS6_4pairIKSD_SE_EEEEEE:
 1028|  4.51k|{
 1029|  4.51k|    Desugarer desugarer(alloc);
 1030|  4.51k|    desugarer.desugarFile(ast, tlas);
 1031|  4.51k|}
_ZN7jsonnet8internal9DesugarerC2EPNS0_9AllocatorEb:
  229|  7.97k|    Desugarer(Allocator *alloc, bool isStdlib = false) : alloc(alloc), isStdlib(isStdlib) {}
_ZN7jsonnet8internal9Desugarer9stdlibASTENSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEE:
  921|  7.56k|    DesugaredObject *stdlibAST(std::string filename) {
  922|       |        // Now, implement the std library by wrapping in a local construct.
  923|  7.56k|        Tokens tokens = jsonnet_lex("std.jsonnet", STD_CODE);
  924|  7.56k|        AST *std_ast = jsonnet_parse(alloc, tokens);
  925|  7.56k|        desugar(std_ast, 0);
  926|  7.56k|        auto *std_obj = dynamic_cast<DesugaredObject *>(std_ast);
  927|  7.56k|        if (std_obj == nullptr) {
  ------------------
  |  Branch (927:13): [True: 0, False: 7.56k]
  ------------------
  928|      0|            std::cerr << "INTERNAL ERROR: std.jsonnet not an object." << std::endl;
  929|      0|            std::abort();
  930|      0|        }
  931|       |
  932|       |        // Bind 'std' builtins that are implemented natively.
  933|  7.56k|        DesugaredObject::Fields &fields = std_obj->fields;
  934|   325k|        for (unsigned long c = 0; c <= max_builtin; ++c) {
  ------------------
  |  Branch (934:35): [True: 317k, False: 7.56k]
  ------------------
  935|   317k|            const auto &decl = jsonnet_builtin_decl(c);
  936|   317k|            auto name = str(decl.name);
  937|   317k|            auto name_utf8 = "std:" + encode_utf8(decl.name);
  938|   317k|            auto fnbody = alloc->makeBuiltinBody(E, name_utf8, [this, &decl]()->Identifiers {
  939|   317k|                Identifiers params;
  940|   317k|                for (const auto &p : decl.params)
  941|   317k|                    params.push_back(this->id(p));
  942|   317k|                return params;
  943|   317k|            });
  944|   317k|            auto field = std::find_if(fields.begin(), fields.end(),
  945|   317k|                [=](const DesugaredObject::Field& f) {
  946|   317k|                    return static_cast<LiteralString*>(f.name)->value == decl.name;
  947|   317k|                });
  948|   317k|            auto fn = make<BuiltinFunction>(E, fnbody);
  949|   317k|            if (field != fields.end()) {
  ------------------
  |  Branch (949:17): [True: 52.9k, False: 264k]
  ------------------
  950|  52.9k|                field->body = fn;
  951|   264k|            } else {
  952|   264k|                fields.emplace_back(ObjectField::HIDDEN, name, fn);
  953|   264k|            }
  954|   317k|        }
  955|  7.56k|        fields.emplace_back(
  956|  7.56k|            ObjectField::HIDDEN, str(U"thisFile"), str(decode_utf8(filename)));
  957|  7.56k|        return std_obj;
  958|  7.56k|    }
_ZN7jsonnet8internal9Desugarer7desugarERPNS0_3ASTEj:
  686|  61.8M|    {
  687|  61.8M|        if (auto *ast = dynamic_cast<Apply *>(ast_)) {
  ------------------
  |  Branch (687:19): [True: 6.15M, False: 55.6M]
  ------------------
  688|  6.15M|            desugar(ast->target, obj_level);
  689|  6.15M|            for (ArgParam &arg : ast->args)
  ------------------
  |  Branch (689:32): [True: 10.1M, False: 6.15M]
  ------------------
  690|  10.1M|                desugar(arg.expr, obj_level);
  691|       |
  692|  55.6M|        } else if (auto *ast = dynamic_cast<ApplyBrace *>(ast_)) {
  ------------------
  |  Branch (692:26): [True: 54.9k, False: 55.6M]
  ------------------
  693|  54.9k|            desugar(ast->left, obj_level);
  694|  54.9k|            desugar(ast->right, obj_level);
  695|  54.9k|            ast_ =
  696|  54.9k|                make<Binary>(ast->location, ast->openFodder, ast->left, EF, BOP_PLUS, ast->right);
  697|       |
  698|  55.6M|        } else if (auto *ast = dynamic_cast<Array *>(ast_)) {
  ------------------
  |  Branch (698:26): [True: 851k, False: 54.7M]
  ------------------
  699|   851k|            for (auto &el : ast->elements)
  ------------------
  |  Branch (699:27): [True: 1.72M, False: 851k]
  ------------------
  700|  1.72M|                desugar(el.expr, obj_level);
  701|       |
  702|  54.7M|        } else if (auto *ast = dynamic_cast<ArrayComprehension *>(ast_)) {
  ------------------
  |  Branch (702:26): [True: 273k, False: 54.5M]
  ------------------
  703|   273k|            for (ComprehensionSpec &spec : ast->specs)
  ------------------
  |  Branch (703:42): [True: 369k, False: 273k]
  ------------------
  704|   369k|                desugar(spec.expr, obj_level);
  705|   273k|            desugar(ast->body, obj_level + 1);
  706|       |
  707|   273k|            ast_ = makeArrayComprehension(ast);
  708|       |
  709|  54.5M|        } else if (auto *ast = dynamic_cast<Assert *>(ast_)) {
  ------------------
  |  Branch (709:26): [True: 310k, False: 54.1M]
  ------------------
  710|   310k|            desugar(ast->cond, obj_level);
  711|   310k|            if (ast->message == nullptr) {
  ------------------
  |  Branch (711:17): [True: 22.9k, False: 287k]
  ------------------
  712|  22.9k|                ast->message = str(U"Assertion failed.");
  713|  22.9k|            }
  714|   310k|            desugar(ast->message, obj_level);
  715|   310k|            desugar(ast->rest, obj_level);
  716|       |
  717|       |            // if cond then rest else error msg
  718|   310k|            AST *branch_false = make<Error>(ast->location, EF, ast->message);
  719|   310k|            ast_ = make<Conditional>(
  720|   310k|                ast->location, ast->openFodder, ast->cond, EF, ast->rest, EF, branch_false);
  721|       |
  722|  54.1M|        } else if (auto *ast = dynamic_cast<Binary *>(ast_)) {
  ------------------
  |  Branch (722:26): [True: 6.58M, False: 47.6M]
  ------------------
  723|  6.58M|            desugar(ast->left, obj_level);
  724|  6.58M|            desugar(ast->right, obj_level);
  725|       |
  726|  6.58M|            bool invert = false;
  727|       |
  728|  6.58M|            switch (ast->op) {
  729|   239k|                case BOP_PERCENT: {
  ------------------
  |  Branch (729:17): [True: 239k, False: 6.34M]
  ------------------
  730|   239k|                    AST *f_mod = make<Index>(
  731|   239k|                        E, EF, std(), EF, false, str(U"mod"), EF, nullptr, EF, nullptr, EF);
  732|   239k|                    ArgParams args = {{ast->left, EF}, {ast->right, EF}};
  733|   239k|                    ast_ = make<Apply>(
  734|   239k|                        ast->location, ast->openFodder, f_mod, EF, args, false, EF, EF, false);
  735|   239k|                } break;
  736|       |
  737|   183k|                case BOP_MANIFEST_UNEQUAL: invert = true;
  ------------------
  |  Branch (737:17): [True: 183k, False: 6.39M]
  ------------------
  738|   183k|                [[fallthrough]];
  739|  1.37M|                case BOP_MANIFEST_EQUAL: {
  ------------------
  |  Branch (739:17): [True: 1.19M, False: 5.39M]
  ------------------
  740|  1.37M|                    ast_ = equals(ast->location, ast->left, ast->right);
  741|  1.37M|                    if (invert)
  ------------------
  |  Branch (741:25): [True: 183k, False: 1.19M]
  ------------------
  742|   183k|                        ast_ = make<Unary>(ast->location, ast->openFodder, UOP_NOT, ast_);
  743|  1.37M|                } break;
  744|       |
  745|  4.96M|                default:;
  ------------------
  |  Branch (745:17): [True: 4.96M, False: 1.61M]
  ------------------
  746|       |                    // Otherwise don't change it.
  747|  6.58M|            }
  748|       |
  749|  47.6M|        } else if (dynamic_cast<const BuiltinFunction *>(ast_)) {
  ------------------
  |  Branch (749:20): [True: 0, False: 47.6M]
  ------------------
  750|       |            // Nothing to do.
  751|       |
  752|  47.6M|        } else if (auto *ast = dynamic_cast<Conditional *>(ast_)) {
  ------------------
  |  Branch (752:26): [True: 2.48M, False: 45.1M]
  ------------------
  753|  2.48M|            desugar(ast->cond, obj_level);
  754|  2.48M|            desugar(ast->branchTrue, obj_level);
  755|  2.48M|            if (ast->branchFalse == nullptr)
  ------------------
  |  Branch (755:17): [True: 30.2k, False: 2.44M]
  ------------------
  756|  30.2k|                ast->branchFalse = null();
  757|  2.48M|            desugar(ast->branchFalse, obj_level);
  758|       |
  759|  45.1M|        } else if (auto *ast = dynamic_cast<Dollar *>(ast_)) {
  ------------------
  |  Branch (759:26): [True: 24.3k, False: 45.1M]
  ------------------
  760|  24.3k|            if (obj_level == 0) {
  ------------------
  |  Branch (760:17): [True: 193, False: 24.1k]
  ------------------
  761|    193|                throw StaticError(ast->location, "No top-level object found.");
  762|    193|            }
  763|  24.1k|            ast_ = var(id(U"$"));
  764|       |
  765|  45.1M|        } else if (auto *ast = dynamic_cast<Error *>(ast_)) {
  ------------------
  |  Branch (765:26): [True: 498k, False: 44.6M]
  ------------------
  766|   498k|            desugar(ast->expr, obj_level);
  767|       |
  768|  44.6M|        } else if (auto *ast = dynamic_cast<Function *>(ast_)) {
  ------------------
  |  Branch (768:26): [True: 154k, False: 44.4M]
  ------------------
  769|   154k|            desugar(ast->body, obj_level);
  770|   154k|            desugarParams(ast->params, obj_level);
  771|       |
  772|  44.4M|        } else if (auto *ast = dynamic_cast<Import *>(ast_)) {
  ------------------
  |  Branch (772:26): [True: 152, False: 44.4M]
  ------------------
  773|       |            // TODO(dcunnin): Abstract this into a template function if it becomes more common.
  774|    152|            AST *file = ast->file;
  775|    152|            desugar(file, obj_level);
  776|    152|            ast->file = dynamic_cast<LiteralString *>(file);
  777|       |
  778|  44.4M|        } else if (auto *ast = dynamic_cast<Importstr *>(ast_)) {
  ------------------
  |  Branch (778:26): [True: 380, False: 44.4M]
  ------------------
  779|       |            // TODO(dcunnin): Abstract this into a template function if it becomes more common.
  780|    380|            AST *file = ast->file;
  781|    380|            desugar(file, obj_level);
  782|    380|            ast->file = dynamic_cast<LiteralString *>(file);
  783|       |
  784|  44.4M|        } else if (auto *ast = dynamic_cast<Importbin *>(ast_)) {
  ------------------
  |  Branch (784:26): [True: 1.67k, False: 44.4M]
  ------------------
  785|       |            // TODO(dcunnin): Abstract this into a template function if it becomes more common.
  786|  1.67k|            AST *file = ast->file;
  787|  1.67k|            desugar(file, obj_level);
  788|  1.67k|            ast->file = dynamic_cast<LiteralString *>(file);
  789|       |
  790|  44.4M|        } else if (auto *ast = dynamic_cast<InSuper *>(ast_)) {
  ------------------
  |  Branch (790:26): [True: 3.71k, False: 44.4M]
  ------------------
  791|  3.71k|            desugar(ast->element, obj_level);
  792|       |
  793|  44.4M|        } else if (auto *ast = dynamic_cast<Index *>(ast_)) {
  ------------------
  |  Branch (793:26): [True: 6.77M, False: 37.6M]
  ------------------
  794|  6.77M|            desugar(ast->target, obj_level);
  795|  6.77M|            if (ast->isSlice) {
  ------------------
  |  Branch (795:17): [True: 140k, False: 6.63M]
  ------------------
  796|   140k|                if (ast->index == nullptr)
  ------------------
  |  Branch (796:21): [True: 19.1k, False: 120k]
  ------------------
  797|  19.1k|                    ast->index = null();
  798|   140k|                desugar(ast->index, obj_level);
  799|       |
  800|   140k|                if (ast->end == nullptr)
  ------------------
  |  Branch (800:21): [True: 77.9k, False: 62.1k]
  ------------------
  801|  77.9k|                    ast->end = null();
  802|   140k|                desugar(ast->end, obj_level);
  803|       |
  804|   140k|                if (ast->step == nullptr)
  ------------------
  |  Branch (804:21): [True: 131k, False: 8.56k]
  ------------------
  805|   131k|                    ast->step = null();
  806|   140k|                desugar(ast->step, obj_level);
  807|       |
  808|   140k|                ast_ = make<Apply>(
  809|   140k|                    ast->location,
  810|   140k|                    EF,
  811|   140k|                    make<Index>(
  812|   140k|                        E, EF, std(), EF, false, str(U"slice"), EF, nullptr, EF, nullptr, EF),
  813|   140k|                    EF,
  814|   140k|                    ArgParams{
  815|   140k|                        {ast->target, EF},
  816|   140k|                        {ast->index, EF},
  817|   140k|                        {ast->end, EF},
  818|   140k|                        {ast->step, EF},
  819|   140k|                    },
  820|   140k|                    false,  // trailing comma
  821|   140k|                    EF,
  822|   140k|                    EF,
  823|   140k|                    false  // tailstrict
  824|   140k|                );
  825|  6.63M|            } else {
  826|  6.63M|                if (ast->id != nullptr) {
  ------------------
  |  Branch (826:21): [True: 5.43M, False: 1.20M]
  ------------------
  827|  5.43M|                    assert(ast->index == nullptr);
  ------------------
  |  Branch (827:21): [True: 5.43M, False: 0]
  ------------------
  828|  5.43M|                    ast->index = str(ast->id->name);
  829|  5.43M|                    ast->id = nullptr;
  830|  5.43M|                }
  831|  6.63M|                desugar(ast->index, obj_level);
  832|  6.63M|            }
  833|       |
  834|  37.6M|        } else if (auto *ast = dynamic_cast<Local *>(ast_)) {
  ------------------
  |  Branch (834:26): [True: 1.83M, False: 35.8M]
  ------------------
  835|  1.83M|            for (auto &bind : ast->binds)
  ------------------
  |  Branch (835:29): [True: 1.94M, False: 1.83M]
  ------------------
  836|  1.94M|                desugar(bind.body, obj_level);
  837|  1.83M|            desugar(ast->body, obj_level);
  838|       |
  839|  1.94M|            for (auto &bind : ast->binds) {
  ------------------
  |  Branch (839:29): [True: 1.94M, False: 1.83M]
  ------------------
  840|  1.94M|                if (bind.functionSugar) {
  ------------------
  |  Branch (840:21): [True: 559k, False: 1.38M]
  ------------------
  841|   559k|                    desugarParams(bind.params, obj_level);
  842|   559k|                    bind.body = make<Function>(ast->location,
  843|   559k|                                               ast->openFodder,
  844|   559k|                                               bind.parenLeftFodder,
  845|   559k|                                               bind.params,
  846|   559k|                                               false,
  847|   559k|                                               bind.parenRightFodder,
  848|   559k|                                               bind.body);
  849|   559k|                    bind.functionSugar = false;
  850|   559k|                    bind.params.clear();
  851|   559k|                }
  852|  1.94M|            }
  853|       |
  854|  35.8M|        } else if (dynamic_cast<const LiteralBoolean *>(ast_)) {
  ------------------
  |  Branch (854:20): [True: 645k, False: 35.1M]
  ------------------
  855|       |            // Nothing to do.
  856|       |
  857|  35.1M|        } else if (dynamic_cast<const LiteralNumber *>(ast_)) {
  ------------------
  |  Branch (857:20): [True: 3.68M, False: 31.5M]
  ------------------
  858|       |            // Nothing to do.
  859|       |
  860|  31.5M|        } else if (auto *ast = dynamic_cast<LiteralString *>(ast_)) {
  ------------------
  |  Branch (860:26): [True: 9.17M, False: 22.3M]
  ------------------
  861|  9.17M|            if ((ast->tokenKind != LiteralString::RAW_DESUGARED) &&
  ------------------
  |  Branch (861:17): [True: 3.66M, False: 5.50M]
  ------------------
  862|  3.66M|                (ast->tokenKind != LiteralString::BLOCK) &&
  ------------------
  |  Branch (862:17): [True: 3.66M, False: 508]
  ------------------
  863|  3.66M|                (ast->tokenKind != LiteralString::VERBATIM_DOUBLE) &&
  ------------------
  |  Branch (863:17): [True: 3.66M, False: 708]
  ------------------
  864|  3.66M|                (ast->tokenKind != LiteralString::VERBATIM_SINGLE)) {
  ------------------
  |  Branch (864:17): [True: 3.66M, False: 190]
  ------------------
  865|  3.66M|                ast->value = jsonnet_string_unescape(ast->location, ast->value);
  866|  3.66M|            }
  867|  9.17M|            ast->tokenKind = LiteralString::RAW_DESUGARED;
  868|  9.17M|            ast->blockIndent.clear();
  869|       |
  870|  22.3M|        } else if (dynamic_cast<const LiteralNull *>(ast_)) {
  ------------------
  |  Branch (870:20): [True: 401k, False: 21.9M]
  ------------------
  871|       |            // Nothing to do.
  872|       |
  873|  21.9M|        } else if (auto *ast = dynamic_cast<DesugaredObject *>(ast_)) {
  ------------------
  |  Branch (873:26): [True: 11.0k, False: 21.9M]
  ------------------
  874|  13.7k|            for (auto &field : ast->fields) {
  ------------------
  |  Branch (874:30): [True: 13.7k, False: 11.0k]
  ------------------
  875|  13.7k|                desugar(field.name, obj_level);
  876|  13.7k|                desugar(field.body, obj_level + 1);
  877|  13.7k|            }
  878|  11.0k|            for (AST *assert : ast->asserts) {
  ------------------
  |  Branch (878:30): [True: 2.59k, False: 11.0k]
  ------------------
  879|  2.59k|                desugar(assert, obj_level + 1);
  880|  2.59k|            }
  881|       |
  882|  21.9M|        } else if (auto *ast = dynamic_cast<Object *>(ast_)) {
  ------------------
  |  Branch (882:26): [True: 376k, False: 21.5M]
  ------------------
  883|   376k|            ast_ = makeObject(ast, obj_level);
  884|       |
  885|  21.5M|        } else if (auto *ast = dynamic_cast<ObjectComprehension *>(ast_)) {
  ------------------
  |  Branch (885:26): [True: 31.6k, False: 21.5M]
  ------------------
  886|  31.6k|            ast_ = makeObjectComprehension(ast, obj_level);
  887|       |
  888|  21.5M|        } else if (auto *ast = dynamic_cast<ObjectComprehensionSimple *>(ast_)) {
  ------------------
  |  Branch (888:26): [True: 2.73k, False: 21.5M]
  ------------------
  889|  2.73k|            desugar(ast->field, obj_level);
  890|  2.73k|            desugar(ast->value, obj_level + 1);
  891|  2.73k|            desugar(ast->array, obj_level);
  892|       |
  893|  21.5M|        } else if (auto *ast = dynamic_cast<Parens *>(ast_)) {
  ------------------
  |  Branch (893:26): [True: 378k, False: 21.1M]
  ------------------
  894|       |            // Strip parens.
  895|   378k|            desugar(ast->expr, obj_level);
  896|   378k|            ast_ = ast->expr;
  897|       |
  898|  21.1M|        } else if (dynamic_cast<const Self *>(ast_)) {
  ------------------
  |  Branch (898:20): [True: 74.6k, False: 21.0M]
  ------------------
  899|       |            // Nothing to do.
  900|       |
  901|  21.0M|        } else if (auto *ast = dynamic_cast<SuperIndex *>(ast_)) {
  ------------------
  |  Branch (901:26): [True: 6.09k, False: 21.0M]
  ------------------
  902|  6.09k|            if (ast->id != nullptr) {
  ------------------
  |  Branch (902:17): [True: 2.46k, False: 3.62k]
  ------------------
  903|  2.46k|                assert(ast->index == nullptr);
  ------------------
  |  Branch (903:17): [True: 2.46k, False: 0]
  ------------------
  904|  2.46k|                ast->index = str(ast->id->name);
  905|  2.46k|                ast->id = nullptr;
  906|  2.46k|            }
  907|  6.09k|            desugar(ast->index, obj_level);
  908|       |
  909|  21.0M|        } else if (auto *ast = dynamic_cast<Unary *>(ast_)) {
  ------------------
  |  Branch (909:26): [True: 522k, False: 20.5M]
  ------------------
  910|   522k|            desugar(ast->expr, obj_level);
  911|       |
  912|  20.5M|        } else if (dynamic_cast<const Var *>(ast_)) {
  ------------------
  |  Branch (912:20): [True: 20.5M, False: 0]
  ------------------
  913|       |            // Nothing to do.
  914|       |
  915|  20.5M|        } else {
  916|      0|            std::cerr << "INTERNAL ERROR: Unknown AST: " << ast_ << std::endl;
  917|      0|            std::abort();
  918|      0|        }
  919|  61.8M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_6BinaryEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTERKSB_NS0_8BinaryOpESF_EEEPT_DpOT0_:
  110|  54.9k|    {
  111|  54.9k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  54.9k|    }
_ZN7jsonnet8internal9Desugarer22makeArrayComprehensionEPNS0_18ArrayComprehensionE:
  417|   273k|    AST* makeArrayComprehension(ArrayComprehension *ast) {
  418|   273k|        int n = ast->specs.size();
  419|   273k|        AST *zero = make<LiteralNumber>(E, EF, "0.0");
  420|   273k|        AST *one = make<LiteralNumber>(E, EF, "1.0");
  421|   273k|        auto *_r = id(U"$r");
  422|   273k|        auto *_l = id(U"$l");
  423|   273k|        std::vector<const Identifier *> _i(n);
  424|   643k|        for (int i = 0; i < n; ++i) {
  ------------------
  |  Branch (424:25): [True: 369k, False: 273k]
  ------------------
  425|   369k|            UStringStream ss;
  426|   369k|            ss << U"$i_" << i;
  427|   369k|            _i[i] = id(ss.str());
  428|   369k|        }
  429|   273k|        std::vector<const Identifier *> _aux(n);
  430|   643k|        for (int i = 0; i < n; ++i) {
  ------------------
  |  Branch (430:25): [True: 369k, False: 273k]
  ------------------
  431|   369k|            UStringStream ss;
  432|   369k|            ss << U"$aux_" << i;
  433|   369k|            _aux[i] = id(ss.str());
  434|   369k|        }
  435|       |
  436|       |        // Build it from the inside out.  We keep wrapping 'in' with more ASTs.
  437|   273k|        assert(ast->specs[0].kind == ComprehensionSpec::FOR);
  ------------------
  |  Branch (437:9): [True: 273k, False: 0]
  ------------------
  438|       |
  439|   273k|        int last_for = n - 1;
  440|   343k|        while (ast->specs[last_for].kind != ComprehensionSpec::FOR)
  ------------------
  |  Branch (440:16): [True: 69.5k, False: 273k]
  ------------------
  441|  69.5k|            last_for--;
  442|       |        // $aux_{last_for}($i_{last_for} + 1, $r + [body])
  443|   273k|        AST *in = make<Apply>(
  444|   273k|            ast->body->location,
  445|   273k|            EF,
  446|   273k|            var(_aux[last_for]),
  447|   273k|            EF,
  448|   273k|            ArgParams{{make<Binary>(E, EF, var(_i[last_for]), EF, BOP_PLUS, one), EF},
  449|   273k|                {make<Binary>(E, EF, var(_r), EF, BOP_PLUS, singleton(ast->body)), EF}},
  450|   273k|            false,  // trailingComma
  451|   273k|            EF,
  452|   273k|            EF,
  453|   273k|            true  // tailstrict
  454|   273k|        );
  455|   643k|        for (int i = n - 1; i >= 0; --i) {
  ------------------
  |  Branch (455:29): [True: 369k, False: 273k]
  ------------------
  456|   369k|            const ComprehensionSpec &spec = ast->specs[i];
  457|   369k|            AST *out;
  458|   369k|            if (i > 0) {
  ------------------
  |  Branch (458:17): [True: 95.8k, False: 273k]
  ------------------
  459|  95.8k|                int prev_for = i - 1;
  460|  23.3M|                while (ast->specs[prev_for].kind != ComprehensionSpec::FOR)
  ------------------
  |  Branch (460:24): [True: 23.2M, False: 95.8k]
  ------------------
  461|  23.2M|                    prev_for--;
  462|       |
  463|       |                // aux_{prev_for}($i_{prev_for} + 1, $r)
  464|  95.8k|                out = make<Apply>(  // False branch.
  465|  95.8k|                    E,
  466|  95.8k|                    EF,
  467|  95.8k|                    var(_aux[prev_for]),
  468|  95.8k|                    EF,
  469|  95.8k|                    ArgParams{{
  470|  95.8k|                        make<Binary>(E, EF, var(_i[prev_for]), EF, BOP_PLUS, one),
  471|  95.8k|                            EF,
  472|  95.8k|                            },
  473|  95.8k|                      {
  474|  95.8k|                        var(_r),
  475|  95.8k|                            EF,
  476|  95.8k|                            }},
  477|  95.8k|                    false,  // trailingComma
  478|  95.8k|                    EF,
  479|  95.8k|                    EF,
  480|  95.8k|                    true  // tailstrict
  481|  95.8k|                );
  482|   273k|            } else {
  483|   273k|                out = var(_r);
  484|   273k|            }
  485|   369k|            switch (spec.kind) {
  ------------------
  |  Branch (485:21): [True: 369k, False: 0]
  ------------------
  486|  69.6k|                case ComprehensionSpec::IF: {
  ------------------
  |  Branch (486:17): [True: 69.6k, False: 299k]
  ------------------
  487|       |                  /*
  488|       |                    if [[[...cond...]]] then
  489|       |                    [[[...in...]]]
  490|       |                    else
  491|       |                    [[[...out...]]]
  492|       |                  */
  493|  69.6k|                    in = make<Conditional>(ast->location,
  494|  69.6k|                                           EF,
  495|  69.6k|                                           spec.expr,
  496|  69.6k|                                           EF,
  497|  69.6k|                                           in,  // True branch.
  498|  69.6k|                                           EF,
  499|  69.6k|                                           out);  // False branch.
  500|  69.6k|                } break;
  501|   299k|                case ComprehensionSpec::FOR: {
  ------------------
  |  Branch (501:17): [True: 299k, False: 69.6k]
  ------------------
  502|       |                    /*
  503|       |                      local $l = [[[...array...]]]
  504|       |                      aux_{i}(i_{i}, r) =
  505|       |                      if i_{i} >= std.length($l) then
  506|       |                      [[[...out...]]]
  507|       |                      else
  508|       |                      local [[[...var...]]] = $l[i_{i}];
  509|       |                      [[[...in...]]];
  510|       |                      if std.type($l) == "array" then
  511|       |                      aux_{i}(0, $r) tailstrict
  512|       |                      else
  513|       |                      error "In comprehension, can only iterate over array..";
  514|       |                    */
  515|   299k|                    in = make<Local>(
  516|   299k|                        ast->location,
  517|   299k|                        EF,
  518|   299k|                        Local::Binds{
  519|   299k|                          bind(_l, spec.expr),  // Need to check expr is an array
  520|   299k|                              bind(_aux[i],
  521|   299k|                                   make<Function>(
  522|   299k|                                       ast->location,
  523|   299k|                                       EF,
  524|   299k|                                       EF,
  525|   299k|                                       ArgParams{{EF, _i[i], EF}, {EF, _r, EF}},
  526|   299k|                                       false,  // trailingComma
  527|   299k|                                       EF,
  528|   299k|                                       make<Conditional>(ast->location,
  529|   299k|                                                         EF,
  530|   299k|                                                         make<Binary>(E,
  531|   299k|                                                                      EF,
  532|   299k|                                                                      var(_i[i]),
  533|   299k|                                                                      EF,
  534|   299k|                                                                      BOP_GREATER_EQ,
  535|   299k|                                                                      length(var(_l))),
  536|   299k|                                                         EF,
  537|   299k|                                                         out,
  538|   299k|                                                         EF,
  539|   299k|                                                         make<Local>(
  540|   299k|                                                             ast->location,
  541|   299k|                                                             EF,
  542|   299k|                                                             singleBind(spec.var,
  543|   299k|                                                                        make<Index>(E,
  544|   299k|                                                                                    EF,
  545|   299k|                                                                                    var(_l),
  546|   299k|                                                                                    EF,
  547|   299k|                                                                                    false,
  548|   299k|                                                                                    var(_i[i]),
  549|   299k|                                                                                    EF,
  550|   299k|                                                                                    nullptr,
  551|   299k|                                                                                    EF,
  552|   299k|                                                                                    nullptr,
  553|   299k|                                                                                    EF)),
  554|   299k|                                                             in))))},
  555|   299k|                        make<Conditional>(
  556|   299k|                            ast->location,
  557|   299k|                            EF,
  558|   299k|                            equals(ast->location, type(var(_l)), str(U"array")),
  559|   299k|                            EF,
  560|   299k|                            make<Apply>(
  561|   299k|                                E,
  562|   299k|                                EF,
  563|   299k|                                var(_aux[i]),
  564|   299k|                                EF,
  565|   299k|                                ArgParams{{zero, EF},
  566|   299k|                                  {
  567|   299k|                                    i == 0 ? make<Array>(
  ------------------
  |  Branch (567:37): [True: 273k, False: 26.2k]
  ------------------
  568|   273k|                                        E, EF, Array::Elements{}, false, EF)
  569|   299k|                                        : static_cast<AST *>(var(_r)),
  570|   299k|                                        EF,
  571|   299k|                                        }},
  572|   299k|                                false,  // trailingComma
  573|   299k|                                EF,
  574|   299k|                                EF,
  575|   299k|                                true),  // tailstrict
  576|   299k|                            EF,
  577|   299k|                            error(ast->location,
  578|   299k|                                  U"In comprehension, can only iterate over array.")));
  579|   299k|                } break;
  580|   369k|            }
  581|   369k|        }
  582|       |
  583|   273k|        return in;
  584|   273k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_13LiteralNumberEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERA4_KcEEEPT_DpOT0_:
  110|   578k|    {
  111|   578k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   578k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ApplyEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_3VarESD_NS7_INS0_8ArgParamENS9_ISG_EEEEbSD_SD_bEEEPT_DpOT0_:
  110|   273k|    {
  111|   273k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   273k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_6BinaryEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS0_8BinaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  110|   369k|    {
  111|   369k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   369k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_6BinaryEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS0_8BinaryOpEPNS0_5ArrayEEEEPT_DpOT0_:
  110|   273k|    {
  111|   273k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   273k|    }
_ZN7jsonnet8internal9Desugarer9singletonEPNS0_3ASTE:
  163|   273k|    {
  164|   273k|        return make<Array>(
  165|   273k|            body->location, EF, Array::Elements{Array::Element(body, EF)}, false, EF);
  166|   273k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ArrayEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_7ElementENS9_ISE_EEEEbSD_EEEPT_DpOT0_:
  110|   273k|    {
  111|   273k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   273k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ApplyEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS8_INS0_8ArgParamENSA_ISH_EEEEbSE_SE_bEEEPT_DpOT0_:
  110|   395k|    {
  111|   395k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   395k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERKPNS0_3ASTESD_RSF_SD_SI_EEEPT_DpOT0_:
  110|  69.6k|    {
  111|  69.6k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  69.6k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5LocalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_4BindENS9_ISE_EEEEPNS0_11ConditionalEEEEPT_DpOT0_:
  110|   299k|    {
  111|   299k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   299k|    }
_ZN7jsonnet8internal9Desugarer4bindEPKNS0_10IdentifierEPNS0_3ASTE:
  153|  1.23M|    {
  154|  1.23M|        return Local::Bind(EF, id, EF, body, false, EF, ArgParams{}, false, EF, EF);
  155|  1.23M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_8FunctionEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEESD_NS7_INS0_8ArgParamENS9_ISE_EEEEbSD_PNS0_11ConditionalEEEEPT_DpOT0_:
  110|   299k|    {
  111|   299k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   299k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_6BinaryESD_RPNS0_3ASTESD_PNS0_5LocalEEEEPT_DpOT0_:
  110|   299k|    {
  111|   299k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   299k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_6BinaryEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_NS0_8BinaryOpEPNS0_5ApplyEEEEPT_DpOT0_:
  110|   299k|    {
  111|   299k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   299k|    }
_ZN7jsonnet8internal9Desugarer6lengthEPNS0_3ASTE:
  199|   299k|    {
  200|   299k|        return stdFunc(U"length", v);
  201|   299k|    }
_ZN7jsonnet8internal9Desugarer7stdFuncERKNSt3__112basic_stringIDiNS2_11char_traitsIDiEENS2_9allocatorIDiEEEEPNS0_3ASTE:
  169|   603k|    {
  170|   603k|        return make<Apply>(
  171|   603k|            v->location,
  172|   603k|            EF,
  173|   603k|            make<Index>(E, EF, std(), EF, false, str(name), EF, nullptr, EF, nullptr, EF),
  174|   603k|            EF,
  175|   603k|            ArgParams{{v, EF}},
  176|   603k|            false,  // trailingComma
  177|   603k|            EF,
  178|   603k|            EF,
  179|   603k|            true  // tailstrict
  180|   603k|        );
  181|   603k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5LocalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_4BindENS9_ISE_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  110|   308k|    {
  111|   308k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   308k|    }
_ZN7jsonnet8internal9Desugarer10singleBindEPKNS0_10IdentifierEPNS0_3ASTE:
  158|   312k|    {
  159|   312k|        return {bind(id, body)};
  160|   312k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bSG_SE_DnSE_DnSE_EEEPT_DpOT0_:
  110|   299k|    {
  111|   299k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   299k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_5ApplyESD_SF_SD_PNS0_5ErrorEEEEPT_DpOT0_:
  110|   299k|    {
  111|   299k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   299k|    }
_ZN7jsonnet8internal9Desugarer4typeEPNS0_3ASTE:
  204|   303k|    {
  205|   303k|        return stdFunc(U"type", v);
  206|   303k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ArrayEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEENS8_INS3_7ElementENSA_ISF_EEEEbSE_EEEPT_DpOT0_:
  110|   273k|    {
  111|   273k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   273k|    }
_ZN7jsonnet8internal9Desugarer5errorERKNS0_13LocationRangeERKNSt3__112basic_stringIDiNS5_11char_traitsIDiEENS5_9allocatorIDiEEEE:
  224|   299k|    {
  225|   299k|        return error(str(loc, msg));
  226|   299k|    }
_ZN7jsonnet8internal9Desugarer5errorEPNS0_3ASTE:
  219|   342k|    {
  220|   342k|        return make<Error>(msg->location, EF, msg);
  221|   342k|    }
_ZN7jsonnet8internal9Desugarer3strERKNS0_13LocationRangeERKNSt3__112basic_stringIDiNS5_11char_traitsIDiEENS5_9allocatorIDiEEEE:
  130|  2.00M|    {
  131|  2.00M|        return make<LiteralString>(loc, EF, s, LiteralString::RAW_DESUGARED, "", "");
  132|  2.00M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_13LiteralStringEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERKNS7_12basic_stringIDiNS7_11char_traitsIDiEENSA_IDiEEEENS3_9TokenKindERA1_KcSP_EEEPT_DpOT0_:
  110|  10.7M|    {
  111|  10.7M|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  10.7M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ErrorEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  110|   653k|    {
  111|   653k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   653k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTERKSB_SF_SH_SF_EEEPT_DpOT0_:
  110|   310k|    {
  111|   310k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   310k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bPNS0_13LiteralStringESE_DnSE_DnSE_EEEPT_DpOT0_:
  110|  2.66M|    {
  111|  2.66M|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  2.66M|    }
_ZN7jsonnet8internal9Desugarer3stdEv:
  145|  2.66M|    {
  146|       |        // In most places, there is a "$std" variable inserted by
  147|       |        // the desugarer. On the standard library itself there isn't,
  148|       |        // so use "std" instead.
  149|  2.66M|        return var(id(isStdlib ? U"std" : U"$std"));
  ------------------
  |  Branch (149:23): [True: 1.19M, False: 1.46M]
  ------------------
  150|  2.66M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ApplyEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTERKSB_RNS7_INS0_8ArgParamENS9_ISI_EEEEbSH_SH_bEEEPT_DpOT0_:
  110|   239k|    {
  111|   239k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   239k|    }
_ZN7jsonnet8internal9Desugarer6equalsERKNS0_13LocationRangeEPNS0_3ASTES6_:
  214|  1.67M|    {
  215|  1.67M|        return stdFunc(loc, U"equals", a, b);
  216|  1.67M|    }
_ZN7jsonnet8internal9Desugarer7stdFuncERKNS0_13LocationRangeERKNSt3__112basic_stringIDiNS5_11char_traitsIDiEENS5_9allocatorIDiEEEEPNS0_3ASTESF_:
  184|  1.67M|    {
  185|  1.67M|        return make<Apply>(
  186|  1.67M|            loc,
  187|  1.67M|            EF,
  188|  1.67M|            make<Index>(E, EF, std(), EF, false, str(name), EF, nullptr, EF, nullptr, EF),
  189|  1.67M|            EF,
  190|  1.67M|            ArgParams{{a, EF}, {b, EF}},
  191|  1.67M|            false,  // trailingComma
  192|  1.67M|            EF,
  193|  1.67M|            EF,
  194|  1.67M|            true  // tailstrict
  195|  1.67M|        );
  196|  1.67M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ApplyEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_5IndexESE_NS8_INS0_8ArgParamENSA_ISH_EEEEbSE_SE_bEEEPT_DpOT0_:
  110|  1.67M|    {
  111|  1.67M|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  1.67M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5UnaryEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS0_7UnaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  110|   183k|    {
  111|   183k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   183k|    }
_ZN7jsonnet8internal9Desugarer4nullEv:
  135|   258k|    {
  136|   258k|        return make<LiteralNull>(E, EF);
  137|   258k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11LiteralNullEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEEEEPT_DpOT0_:
  110|   258k|    {
  111|   258k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   258k|    }
_ZN7jsonnet8internal9Desugarer3varEPKNS0_10IdentifierE:
  140|  6.00M|    {
  141|  6.00M|        return make<Var>(E, EF, ident);
  142|  6.00M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_3VarEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERPKNS0_10IdentifierEEEEPT_DpOT0_:
  110|  6.00M|    {
  111|  6.00M|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  6.00M|    }
_ZN7jsonnet8internal9Desugarer2idERKNSt3__112basic_stringIDiNS2_11char_traitsIDiEENS2_9allocatorIDiEEEE:
  120|  4.51M|    {
  121|  4.51M|        return alloc->makeIdentifier(s);
  122|  4.51M|    }
_ZN7jsonnet8internal9Desugarer13desugarParamsERNSt3__16vectorINS0_8ArgParamENS2_9allocatorIS4_EEEEj:
  232|  2.63M|    {
  233|  3.24M|        for (auto &param : params) {
  ------------------
  |  Branch (233:26): [True: 3.24M, False: 2.63M]
  ------------------
  234|  3.24M|            if (param.expr) {
  ------------------
  |  Branch (234:17): [True: 159k, False: 3.08M]
  ------------------
  235|       |                // Default arg.
  236|   159k|                desugar(param.expr, obj_level);
  237|   159k|            }
  238|  3.24M|        }
  239|  2.63M|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ApplyEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_5IndexESD_NS7_INS0_8ArgParamENS9_ISG_EEEEbSD_SD_bEEEPT_DpOT0_:
  110|   743k|    {
  111|   743k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   743k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_8FunctionEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEESC_RNS7_INS0_8ArgParamENS9_ISD_EEEEbSC_RPNS0_3ASTEEEEPT_DpOT0_:
  110|   559k|    {
  111|   559k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   559k|    }
_ZN7jsonnet8internal9Desugarer10makeObjectEPNS0_6ObjectEj:
  586|   376k|    AST* makeObject(Object *ast, unsigned obj_level) {
  587|       |        // Hidden variable to allow outer/top binding.
  588|   376k|        if (obj_level == 0) {
  ------------------
  |  Branch (588:13): [True: 20.3k, False: 356k]
  ------------------
  589|  20.3k|            const Identifier *hidden_var = id(U"$");
  590|  20.3k|            auto *body = make<Self>(E, EF);
  591|  20.3k|            ast->fields.push_back(ObjectField::Local(EF, EF, hidden_var, EF, body, EF));
  592|  20.3k|        }
  593|       |
  594|   376k|        SuperVars svs = desugarFields(ast, ast->fields, obj_level);
  595|       |
  596|   376k|        DesugaredObject::Fields new_fields;
  597|   376k|        ASTs new_asserts;
  598|  1.81M|        for (const ObjectField &field : ast->fields) {
  ------------------
  |  Branch (598:39): [True: 1.81M, False: 376k]
  ------------------
  599|  1.81M|            if (field.kind == ObjectField::ASSERT) {
  ------------------
  |  Branch (599:17): [True: 42.9k, False: 1.77M]
  ------------------
  600|  42.9k|                new_asserts.push_back(field.expr2);
  601|  1.77M|            } else if (field.kind == ObjectField::FIELD_EXPR) {
  ------------------
  |  Branch (601:24): [True: 1.77M, False: 0]
  ------------------
  602|  1.77M|                new_fields.emplace_back(field.hide, field.expr1, field.expr2);
  603|  1.77M|            } else {
  604|      0|                std::cerr << "INTERNAL ERROR: field should have been desugared: " << field.kind
  605|      0|                          << std::endl;
  606|      0|            }
  607|  1.81M|        }
  608|       |
  609|   376k|        AST* retval = make<DesugaredObject>(ast->location, new_asserts, new_fields);
  610|   376k|        if (svs.size() > 0) {
  ------------------
  |  Branch (610:13): [True: 1.57k, False: 375k]
  ------------------
  611|  1.57k|            Local::Binds binds;
  612|   214k|            for (const auto &pair : svs) {
  ------------------
  |  Branch (612:35): [True: 214k, False: 1.57k]
  ------------------
  613|   214k|                if (pair.second == nullptr) {
  ------------------
  |  Branch (613:21): [True: 533, False: 213k]
  ------------------
  614|       |                    // Self binding
  615|    533|                    binds.push_back(bind(pair.first, make<Self>(E, EF)));
  616|   213k|                } else {
  617|       |                    // Super binding
  618|   213k|                    binds.push_back(bind(pair.first, pair.second));
  619|   213k|                }
  620|   214k|            }
  621|  1.57k|            retval = make<Local>(ast->location, EF, binds, retval);
  622|  1.57k|        }
  623|       |
  624|   376k|        return retval;
  625|   376k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_4SelfEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEEEEPT_DpOT0_:
  110|  21.5k|    {
  111|  21.5k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  21.5k|    }
_ZN7jsonnet8internal9Desugarer13desugarFieldsEPNS0_3ASTERNSt3__16vectorINS0_11ObjectFieldENS4_9allocatorIS6_EEEEj:
  246|   408k|    {
  247|       |        // Desugar children
  248|  1.91M|        for (auto &field : fields) {
  ------------------
  |  Branch (248:26): [True: 1.91M, False: 408k]
  ------------------
  249|  1.91M|            if (field.expr1 != nullptr)
  ------------------
  |  Branch (249:17): [True: 101k, False: 1.81M]
  ------------------
  250|   101k|                desugar(field.expr1, obj_level);
  251|  1.91M|            desugar(field.expr2, obj_level + 1);
  252|  1.91M|            if (field.expr3 != nullptr)
  ------------------
  |  Branch (252:17): [True: 42.0k, False: 1.87M]
  ------------------
  253|  42.0k|                desugar(field.expr3, obj_level + 1);
  254|  1.91M|            desugarParams(field.params, obj_level + 1);
  255|  1.91M|        }
  256|       |
  257|       |        // Simplify asserts
  258|  1.91M|        for (auto &field : fields) {
  ------------------
  |  Branch (258:26): [True: 1.91M, False: 408k]
  ------------------
  259|  1.91M|            if (field.kind != ObjectField::ASSERT)
  ------------------
  |  Branch (259:17): [True: 1.87M, False: 42.9k]
  ------------------
  260|  1.87M|                continue;
  261|  42.9k|            AST *msg = field.expr3;
  262|  42.9k|            field.expr3 = nullptr;
  263|  42.9k|            if (msg == nullptr) {
  ------------------
  |  Branch (263:17): [True: 832, False: 42.0k]
  ------------------
  264|       |                // The location is what appears in the stacktrace.
  265|    832|                msg = str(field.expr2->location, U"Object assertion failed.");
  266|    832|            }
  267|       |
  268|       |            // if expr2 then true else error msg
  269|  42.9k|            field.expr2 = make<Conditional>(field.expr2->location,
  270|  42.9k|                                            EF,
  271|  42.9k|                                            field.expr2,
  272|  42.9k|                                            EF,
  273|  42.9k|                                            make<LiteralBoolean>(E, EF, true),
  274|  42.9k|                                            EF,
  275|  42.9k|                                            error(msg));
  276|  42.9k|        }
  277|       |
  278|       |        // Remove methods
  279|  1.91M|        for (auto &field : fields) {
  ------------------
  |  Branch (279:26): [True: 1.91M, False: 408k]
  ------------------
  280|  1.91M|            if (!field.methodSugar)
  ------------------
  |  Branch (280:17): [True: 972k, False: 946k]
  ------------------
  281|   972k|                continue;
  282|   946k|            field.expr2 = make<Function>(field.expr2->location,
  283|   946k|                                         EF,
  284|   946k|                                         field.fodderL,
  285|   946k|                                         field.params,
  286|   946k|                                         field.trailingComma,
  287|   946k|                                         field.fodderR,
  288|   946k|                                         field.expr2);
  289|   946k|            field.methodSugar = false;
  290|   946k|            field.params.clear();
  291|   946k|        }
  292|       |
  293|       |        // Remove object-level locals
  294|   408k|        auto copy = fields;
  295|   408k|        fields.clear();
  296|   408k|        Local::Binds binds;
  297|  1.91M|        for (auto &local : copy) {
  ------------------
  |  Branch (297:26): [True: 1.91M, False: 408k]
  ------------------
  298|  1.91M|            if (local.kind != ObjectField::LOCAL)
  ------------------
  |  Branch (298:17): [True: 1.84M, False: 73.9k]
  ------------------
  299|  1.84M|                continue;
  300|  73.9k|            binds.push_back(bind(local.id, local.expr2));
  301|  73.9k|        }
  302|  1.91M|        for (auto &field : copy) {
  ------------------
  |  Branch (302:26): [True: 1.91M, False: 408k]
  ------------------
  303|  1.91M|            if (field.kind == ObjectField::LOCAL)
  ------------------
  |  Branch (303:17): [True: 73.9k, False: 1.84M]
  ------------------
  304|  73.9k|                continue;
  305|  1.84M|            if (!binds.empty())
  ------------------
  |  Branch (305:17): [True: 1.00M, False: 840k]
  ------------------
  306|  1.00M|                field.expr2 = make<Local>(field.expr2->location, EF, binds, field.expr2);
  307|  1.84M|            fields.push_back(field);
  308|  1.84M|        }
  309|       |
  310|       |        // Change all to FIELD_EXPR
  311|  1.84M|        for (auto &field : fields) {
  ------------------
  |  Branch (311:26): [True: 1.84M, False: 408k]
  ------------------
  312|  1.84M|            switch (field.kind) {
  ------------------
  |  Branch (312:21): [True: 1.84M, False: 0]
  ------------------
  313|  42.9k|                case ObjectField::ASSERT:
  ------------------
  |  Branch (313:17): [True: 42.9k, False: 1.80M]
  ------------------
  314|       |                    // Nothing to do.
  315|  42.9k|                    break;
  316|       |
  317|  1.70M|                case ObjectField::FIELD_ID:
  ------------------
  |  Branch (317:17): [True: 1.70M, False: 143k]
  ------------------
  318|  1.70M|                    field.expr1 = str(field.idLocation, field.id->name);
  319|  1.70M|                    field.kind = ObjectField::FIELD_EXPR;
  320|  1.70M|                    break;
  321|       |
  322|  35.6k|                case ObjectField::FIELD_EXPR:
  ------------------
  |  Branch (322:17): [True: 35.6k, False: 1.80M]
  ------------------
  323|       |                    // Nothing to do.
  324|  35.6k|                    break;
  325|       |
  326|  65.3k|                case ObjectField::FIELD_STR:
  ------------------
  |  Branch (326:17): [True: 65.3k, False: 1.77M]
  ------------------
  327|       |                    // Just set the flag.
  328|  65.3k|                    field.kind = ObjectField::FIELD_EXPR;
  329|  65.3k|                    break;
  330|       |
  331|      0|                case ObjectField::LOCAL:
  ------------------
  |  Branch (331:17): [True: 0, False: 1.84M]
  ------------------
  332|      0|                    std::cerr << "Locals should be removed by now." << std::endl;
  333|      0|                    abort();
  334|  1.84M|            }
  335|  1.84M|        }
  336|       |
  337|       |        /** Replaces all occurrences of self, super[f] and e in super with variables.
  338|       |         *
  339|       |         * Returns all variables and original expressions via super_vars.
  340|       |         */
  341|   408k|        class SubstituteSelfSuper : public CompilerPass {
  342|   408k|            Desugarer *desugarer;
  343|   408k|            SuperVars &superVars;
  344|   408k|            unsigned &counter;
  345|   408k|            const Identifier *newSelf;
  346|       |
  347|   408k|           public:
  348|   408k|            SubstituteSelfSuper(Desugarer *desugarer, SuperVars &super_vars, unsigned &counter)
  349|   408k|                : CompilerPass(*desugarer->alloc),
  350|   408k|                  desugarer(desugarer),
  351|   408k|                  superVars(super_vars),
  352|   408k|                  counter(counter),
  353|   408k|                  newSelf(nullptr)
  354|   408k|            {
  355|   408k|            }
  356|   408k|            void visitExpr(AST *&expr)
  357|   408k|            {
  358|   408k|                if (dynamic_cast<Self *>(expr)) {
  359|   408k|                    if (newSelf == nullptr) {
  360|   408k|                        newSelf = desugarer->id(U"$outer_self");
  361|   408k|                        superVars.emplace_back(newSelf, nullptr);
  362|   408k|                    }
  363|   408k|                    expr = alloc.make<Var>(expr->location, expr->openFodder, newSelf);
  364|   408k|                } else if (auto *super_index = dynamic_cast<SuperIndex *>(expr)) {
  365|   408k|                    UStringStream ss;
  366|   408k|                    ss << U"$outer_super_index" << (counter++);
  367|   408k|                    const Identifier *super_var = desugarer->id(ss.str());
  368|       |                    // Desugaring of expr should already have occurred.
  369|   408k|                    assert(super_index->index != nullptr);
  370|       |                    // Re-use super_index since we're replacing it here.
  371|   408k|                    superVars.emplace_back(super_var, super_index);
  372|   408k|                    expr = alloc.make<Var>(expr->location, expr->openFodder, super_var);
  373|   408k|                } else if (auto *in_super = dynamic_cast<InSuper *>(expr)) {
  374|   408k|                    UStringStream ss;
  375|   408k|                    ss << U"$outer_in_super" << (counter++);
  376|   408k|                    const Identifier *in_super_var = desugarer->id(ss.str());
  377|       |                    // Re-use in_super since we're replacing it here.
  378|   408k|                    superVars.emplace_back(in_super_var, in_super);
  379|   408k|                    expr = alloc.make<Var>(expr->location, expr->openFodder, in_super_var);
  380|   408k|                }
  381|   408k|                CompilerPass::visitExpr(expr);
  382|   408k|            }
  383|   408k|        };
  384|       |
  385|   408k|        SuperVars super_vars;
  386|   408k|        unsigned counter = 0;
  387|       |
  388|       |        // Remove +:
  389|  1.84M|        for (auto &field : fields) {
  ------------------
  |  Branch (389:26): [True: 1.84M, False: 408k]
  ------------------
  390|  1.84M|            if (!field.superSugar)
  ------------------
  |  Branch (390:17): [True: 1.83M, False: 9.91k]
  ------------------
  391|  1.83M|                continue;
  392|       |            // We have to bind self/super from expr1 outside the class, as we copy the expression
  393|       |            // into the field body.
  394|       |            // Clone it so that we maintain the AST as a tree.
  395|  9.91k|            AST *index = clone(field.expr1);
  396|       |            // This will remove self/super.
  397|  9.91k|            SubstituteSelfSuper(this, super_vars, counter).expr(index);
  398|  9.91k|            field.expr2 = make<Conditional>(
  399|  9.91k|                ast->location,
  400|  9.91k|                EF,
  401|  9.91k|                make<InSuper>(ast->location, EF, index, EF, EF),
  402|  9.91k|                EF,
  403|  9.91k|                make<Binary>(ast->location,
  404|  9.91k|                             EF,
  405|  9.91k|                             make<SuperIndex>(ast->location, EF, EF, clone(index), EF, nullptr),
  406|  9.91k|                             EF,
  407|  9.91k|                             BOP_PLUS,
  408|  9.91k|                             field.expr2),
  409|  9.91k|                EF,
  410|  9.91k|                clone(field.expr2));
  411|  9.91k|            field.superSugar = false;
  412|  9.91k|        }
  413|       |
  414|   408k|        return super_vars;
  415|   408k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTESD_PNS0_14LiteralBooleanESD_PNS0_5ErrorEEEEPT_DpOT0_:
  110|  42.9k|    {
  111|  42.9k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  42.9k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_14LiteralBooleanEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEbEEEPT_DpOT0_:
  110|  42.9k|    {
  111|  42.9k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  42.9k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_8FunctionEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERSB_RNS7_INS0_8ArgParamENS9_ISF_EEEERbSE_RPNS0_3ASTEEEEPT_DpOT0_:
  110|   946k|    {
  111|   946k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   946k|    }
_ZN7jsonnet8internal9Desugarer5cloneEPNS0_3ASTE:
  115|  29.7k|    {
  116|  29.7k|        return clone_ast(*alloc, ast);
  117|  29.7k|    }
_ZZN7jsonnet8internal9Desugarer13desugarFieldsEPNS0_3ASTERNSt3__16vectorINS0_11ObjectFieldENS4_9allocatorIS6_EEEEjEN19SubstituteSelfSuperC2EPS1_RNS5_INS4_4pairIPKNS0_10IdentifierES3_EENS7_ISH_EEEERj:
  349|  9.91k|                : CompilerPass(*desugarer->alloc),
  350|  9.91k|                  desugarer(desugarer),
  351|  9.91k|                  superVars(super_vars),
  352|  9.91k|                  counter(counter),
  353|  9.91k|                  newSelf(nullptr)
  354|  9.91k|            {
  355|  9.91k|            }
_ZZN7jsonnet8internal9Desugarer13desugarFieldsEPNS0_3ASTERNSt3__16vectorINS0_11ObjectFieldENS4_9allocatorIS6_EEEEjEN19SubstituteSelfSuper9visitExprERS3_:
  357|  1.29M|            {
  358|  1.29M|                if (dynamic_cast<Self *>(expr)) {
  ------------------
  |  Branch (358:21): [True: 4.61k, False: 1.29M]
  ------------------
  359|  4.61k|                    if (newSelf == nullptr) {
  ------------------
  |  Branch (359:25): [True: 677, False: 3.94k]
  ------------------
  360|    677|                        newSelf = desugarer->id(U"$outer_self");
  361|    677|                        superVars.emplace_back(newSelf, nullptr);
  362|    677|                    }
  363|  4.61k|                    expr = alloc.make<Var>(expr->location, expr->openFodder, newSelf);
  364|  1.29M|                } else if (auto *super_index = dynamic_cast<SuperIndex *>(expr)) {
  ------------------
  |  Branch (364:34): [True: 114k, False: 1.17M]
  ------------------
  365|   114k|                    UStringStream ss;
  366|   114k|                    ss << U"$outer_super_index" << (counter++);
  367|   114k|                    const Identifier *super_var = desugarer->id(ss.str());
  368|       |                    // Desugaring of expr should already have occurred.
  369|   114k|                    assert(super_index->index != nullptr);
  ------------------
  |  Branch (369:21): [True: 114k, False: 0]
  ------------------
  370|       |                    // Re-use super_index since we're replacing it here.
  371|   114k|                    superVars.emplace_back(super_var, super_index);
  372|   114k|                    expr = alloc.make<Var>(expr->location, expr->openFodder, super_var);
  373|  1.17M|                } else if (auto *in_super = dynamic_cast<InSuper *>(expr)) {
  ------------------
  |  Branch (373:34): [True: 107k, False: 1.06M]
  ------------------
  374|   107k|                    UStringStream ss;
  375|   107k|                    ss << U"$outer_in_super" << (counter++);
  376|   107k|                    const Identifier *in_super_var = desugarer->id(ss.str());
  377|       |                    // Re-use in_super since we're replacing it here.
  378|   107k|                    superVars.emplace_back(in_super_var, in_super);
  379|   107k|                    expr = alloc.make<Var>(expr->location, expr->openFodder, in_super_var);
  380|   107k|                }
  381|  1.29M|                CompilerPass::visitExpr(expr);
  382|  1.29M|            }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_7InSuperESD_PNS0_6BinaryESD_PNS0_3ASTEEEEPT_DpOT0_:
  110|  9.91k|    {
  111|  9.91k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  9.91k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_7InSuperEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERPNS0_3ASTESD_SD_EEEPT_DpOT0_:
  110|  9.91k|    {
  111|  9.91k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  9.91k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_6BinaryEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_10SuperIndexESD_NS0_8BinaryOpERPNS0_3ASTEEEEPT_DpOT0_:
  110|  9.91k|    {
  111|  9.91k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  9.91k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_10SuperIndexEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEESD_PNS0_3ASTESD_DnEEEPT_DpOT0_:
  110|  9.91k|    {
  111|  9.91k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  9.91k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_15DesugaredObjectEJRNS0_13LocationRangeERNSt3__14listIPNS0_3ASTENS6_9allocatorIS9_EEEERNS6_6vectorINS3_5FieldENSA_ISF_EEEEEEEPT_DpOT0_:
  110|   376k|    {
  111|   376k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   376k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5LocalEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERNS7_INS3_4BindENS9_ISE_EEEERPNS0_3ASTEEEEPT_DpOT0_:
  110|  1.03M|    {
  111|  1.03M|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  1.03M|    }
_ZN7jsonnet8internal9Desugarer23makeObjectComprehensionEPNS0_19ObjectComprehensionEj:
  627|  31.6k|    AST* makeObjectComprehension(ObjectComprehension *ast, unsigned obj_level) {
  628|       |        // Hidden variable to allow outer/top binding.
  629|  31.6k|        if (obj_level == 0) {
  ------------------
  |  Branch (629:13): [True: 679, False: 30.9k]
  ------------------
  630|    679|            const Identifier *hidden_var = id(U"$");
  631|    679|            auto *body = make<Self>(E, EF);
  632|    679|            ast->fields.push_back(ObjectField::Local(EF, EF, hidden_var, EF, body, EF));
  633|    679|        }
  634|       |
  635|  31.6k|        SuperVars svs = desugarFields(ast, ast->fields, obj_level);
  636|       |
  637|  31.6k|        AST *field = ast->fields.front().expr1;
  638|  31.6k|        AST *value = ast->fields.front().expr2;
  639|       |
  640|       |        /*  {
  641|       |            [arr[0]]: local x = arr[1], y = arr[2], z = arr[3]; val_expr
  642|       |            for arr in [ [key_expr, x, y, z] for ...  ]
  643|       |            }
  644|       |        */
  645|  31.6k|        auto *_arr = id(U"$arr");
  646|  31.6k|        AST *zero = make<LiteralNumber>(E, EF, "0.0");
  647|  31.6k|        int counter = 1;
  648|  31.6k|        Local::Binds binds;
  649|  31.6k|        Array::Elements arr_e{Array::Element(field, EF)};
  650|  43.5k|        for (ComprehensionSpec &spec : ast->specs) {
  ------------------
  |  Branch (650:38): [True: 43.5k, False: 31.6k]
  ------------------
  651|  43.5k|            if (spec.kind == ComprehensionSpec::FOR) {
  ------------------
  |  Branch (651:17): [True: 35.2k, False: 8.29k]
  ------------------
  652|  35.2k|                std::stringstream num;
  653|  35.2k|                num << counter++;
  654|  35.2k|                binds.push_back(bind(spec.var,
  655|  35.2k|                                     make<Index>(E,
  656|  35.2k|                                                 EF,
  657|  35.2k|                                                 var(_arr),
  658|  35.2k|                                                 EF,
  659|  35.2k|                                                 false,
  660|  35.2k|                                                 make<LiteralNumber>(E, EF, num.str()),
  661|  35.2k|                                                 EF,
  662|  35.2k|                                                 nullptr,
  663|  35.2k|                                                 EF,
  664|  35.2k|                                                 nullptr,
  665|  35.2k|                                                 EF)));
  666|  35.2k|                arr_e.emplace_back(var(spec.var), EF);
  667|  35.2k|            }
  668|  43.5k|        }
  669|  31.6k|        AST *arr = make<ArrayComprehension>(ast->location,
  670|  31.6k|                                            EF,
  671|  31.6k|                                            make<Array>(ast->location, EF, arr_e, false, EF),
  672|  31.6k|                                            EF,
  673|  31.6k|                                            false,
  674|  31.6k|                                            ast->specs,
  675|  31.6k|                                            EF);
  676|  31.6k|        desugar(arr, obj_level);
  677|  31.6k|        return make<ObjectComprehensionSimple>(
  678|  31.6k|            ast->location,
  679|  31.6k|            make<Index>(E, EF, var(_arr), EF, false, zero, EF, nullptr, EF, nullptr, EF),
  680|  31.6k|            make<Local>(ast->location, EF, binds, value),
  681|  31.6k|            _arr,
  682|  31.6k|            arr);
  683|  31.6k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bPNS0_13LiteralNumberESE_DnSE_DnSE_EEEPT_DpOT0_:
  110|  35.2k|    {
  111|  35.2k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  35.2k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_13LiteralNumberEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEENS7_12basic_stringIcNS7_11char_traitsIcEENSA_IcEEEEEEEPT_DpOT0_:
  110|  35.2k|    {
  111|  35.2k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  35.2k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_18ArrayComprehensionEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_5ArrayESD_bRNS7_INS0_17ComprehensionSpecENS9_ISG_EEEESD_EEEPT_DpOT0_:
  110|  31.6k|    {
  111|  31.6k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  31.6k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ArrayEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEERNS7_INS3_7ElementENS9_ISE_EEEEbSD_EEEPT_DpOT0_:
  110|  31.6k|    {
  111|  31.6k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  31.6k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_25ObjectComprehensionSimpleEJRNS0_13LocationRangeEPNS0_5IndexEPNS0_5LocalERPKNS0_10IdentifierERPNS0_3ASTEEEEPT_DpOT0_:
  110|  31.6k|    {
  111|  31.6k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  31.6k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5IndexEJRKNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_3VarESE_bRPNS0_3ASTESE_DnSE_DnSE_EEEPT_DpOT0_:
  110|  31.6k|    {
  111|  31.6k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  31.6k|    }
_ZN7jsonnet8internal9Desugarer3strERKNSt3__112basic_stringIDiNS2_11char_traitsIDiEENS2_9allocatorIDiEEEE:
  125|  8.75M|    {
  126|  8.75M|        return make<LiteralString>(E, EF, s, LiteralString::RAW_DESUGARED, "", "");
  127|  8.75M|    }
_ZZN7jsonnet8internal9Desugarer9stdlibASTENSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEEENKUlvE_clEv:
  938|   172k|            auto fnbody = alloc->makeBuiltinBody(E, name_utf8, [this, &decl]()->Identifiers {
  939|   172k|                Identifiers params;
  940|   172k|                for (const auto &p : decl.params)
  ------------------
  |  Branch (940:36): [True: 254k, False: 172k]
  ------------------
  941|   254k|                    params.push_back(this->id(p));
  942|   172k|                return params;
  943|   172k|            });
_ZZN7jsonnet8internal9Desugarer9stdlibASTENSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEEENKUlRKNS0_15DesugaredObject5FieldEE_clESC_:
  945|  38.8M|                [=](const DesugaredObject::Field& f) {
  946|  38.8M|                    return static_cast<LiteralString*>(f.name)->value == decl.name;
  947|  38.8M|                });
_ZN7jsonnet8internal9Desugarer4makeINS0_15BuiltinFunctionEJRKNS0_13LocationRangeERPKNS0_19BuiltinFunctionBodyEEEEPT_DpOT0_:
  110|   317k|    {
  111|   317k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|   317k|    }
_ZN7jsonnet8internal9Desugarer11desugarFileERPNS0_3ASTEPNSt3__13mapINS5_12basic_stringIcNS5_11char_traitsIcEENS5_9allocatorIcEEEENS0_5VmExtENS5_4lessISC_EENSA_INS5_4pairIKSC_SD_EEEEEE:
  961|  4.51k|    {
  962|  4.51k|        desugar(ast, 0);
  963|       |
  964|  4.51k|        DesugaredObject *std_obj = stdlibAST(ast->location.file);
  965|       |
  966|  4.51k|        std::vector<std::string> empty;
  967|  4.51k|        auto line_end_blank = Fodder{{FodderElement::LINE_END, 1, 0, empty}};
  968|  4.51k|        auto line_end = Fodder{{FodderElement::LINE_END, 0, 0, empty}};
  969|       |
  970|       |        // local body = ast;
  971|       |        // if std.type(body) == "function") then
  972|       |        //     body(tlas...)
  973|       |        // else
  974|       |        //     body
  975|  4.51k|        if (tlas != nullptr) {
  ------------------
  |  Branch (975:13): [True: 4.10k, False: 416]
  ------------------
  976|  4.10k|            LocationRange tla_loc("Top-level function");
  977|  4.10k|            ArgParams args;
  978|  4.10k|            for (const auto &pair : *tlas) {
  ------------------
  |  Branch (978:35): [True: 0, False: 4.10k]
  ------------------
  979|      0|                AST *expr;
  980|      0|                if (pair.second.isCode) {
  ------------------
  |  Branch (980:21): [True: 0, False: 0]
  ------------------
  981|      0|                    Tokens tokens = jsonnet_lex("tla:" + pair.first, pair.second.data.c_str());
  982|      0|                    expr = jsonnet_parse(alloc, tokens);
  983|      0|                    desugar(expr, 0);
  984|      0|                } else {
  985|      0|                    expr = str(decode_utf8(pair.second.data));
  986|      0|                }
  987|       |                // Add them as named arguments, so order does not matter.
  988|      0|                args.emplace_back(EF, id(decode_utf8(pair.first)), EF, expr, EF);
  989|      0|            }
  990|  4.10k|            const Identifier *body = id(U"top_level");
  991|  4.10k|            ast =
  992|  4.10k|                make<Local>(ast->location,
  993|  4.10k|                            line_end_blank,
  994|  4.10k|                            singleBind(body, ast),
  995|  4.10k|                            make<Conditional>(E,
  996|  4.10k|                                              line_end,
  997|  4.10k|                                              primitiveEquals(E, type(var(body)), str(U"function")),
  998|  4.10k|                                              EF,
  999|  4.10k|                                              make<Apply>(tla_loc,
 1000|  4.10k|                                                          EF,
 1001|  4.10k|                                                          make<Var>(E, line_end, body),
 1002|  4.10k|                                                          EF,
 1003|  4.10k|                                                          args,
 1004|  4.10k|                                                          false,  // trailing comma
 1005|  4.10k|                                                          EF,
 1006|  4.10k|                                                          EF,
 1007|  4.10k|                                                          false  // tailstrict
 1008|  4.10k|                                                          ),
 1009|  4.10k|                                              line_end,
 1010|  4.10k|                                              make<Var>(E, line_end, body)));
 1011|  4.10k|        }
 1012|       |
 1013|       |        // local $std = (std.jsonnet stuff); std = $std; ast
 1014|       |        // The standard library is bound to $std, which cannot be overriden,
 1015|       |        // so redefining std won't break expressions that desugar to calls
 1016|       |        // to standard library functions.
 1017|  4.51k|        ast = make<Local>(ast->location, EF, singleBind(id(U"std"), std()), ast);
 1018|  4.51k|        ast = make<Local>(ast->location, EF, singleBind(id(U"$std"), std_obj), ast);
 1019|  4.51k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5LocalEJRNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEENS7_INS3_4BindENS9_ISD_EEEEPNS0_11ConditionalEEEEPT_DpOT0_:
  110|  4.10k|    {
  111|  4.10k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  4.10k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_11ConditionalEJRKNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEEPNS0_5ApplyERKSC_SF_SD_PNS0_3VarEEEEPT_DpOT0_:
  110|  4.10k|    {
  111|  4.10k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  4.10k|    }
_ZN7jsonnet8internal9Desugarer15primitiveEqualsERKNS0_13LocationRangeEPNS0_3ASTES6_:
  209|  4.10k|    {
  210|  4.10k|        return stdFunc(loc, U"primitiveEquals", a, b);
  211|  4.10k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_5ApplyEJRNS0_13LocationRangeERKNSt3__16vectorINS0_13FodderElementENS6_9allocatorIS8_EEEEPNS0_3VarESD_RNS7_INS0_8ArgParamENS9_ISG_EEEEbSD_SD_bEEEPT_DpOT0_:
  110|  4.10k|    {
  111|  4.10k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  4.10k|    }
_ZN7jsonnet8internal9Desugarer4makeINS0_3VarEJRKNS0_13LocationRangeERNSt3__16vectorINS0_13FodderElementENS7_9allocatorIS9_EEEERPKNS0_10IdentifierEEEEPT_DpOT0_:
  110|  8.20k|    {
  111|  8.20k|        return alloc->make<T>(std::forward<Args>(args)...);
  112|  8.20k|    }

_ZN7jsonnet8internal7FmtOptsC2Ev:
   37|  7.42k|        : stringStyle('s'),
   38|  7.42k|          commentStyle('s'),
   39|  7.42k|          indent(2),
   40|  7.42k|          maxBlankLines(2),
   41|  7.42k|          padArrays(false),
   42|  7.42k|          padObjects(true),
   43|  7.42k|          stripComments(false),
   44|  7.42k|          stripAllButComments(false),
   45|  7.42k|          stripEverything(false),
   46|  7.42k|          prettyFieldNames(true),
   47|  7.42k|          sortImports(true)
   48|  7.42k|    {
   49|  7.42k|    }

_ZN7jsonnet8internal26allowed_at_end_of_operatorEc:
  174|  6.48M|bool allowed_at_end_of_operator(char c) {
  175|  6.48M|    switch (c) {
  ------------------
  |  Branch (175:13): [True: 3.25M, False: 3.22M]
  ------------------
  176|  2.30M|        case '+':
  ------------------
  |  Branch (176:9): [True: 2.30M, False: 4.17M]
  ------------------
  177|  2.34M|        case '-':
  ------------------
  |  Branch (177:9): [True: 38.3k, False: 6.44M]
  ------------------
  178|  2.80M|        case '~':
  ------------------
  |  Branch (178:9): [True: 459k, False: 6.02M]
  ------------------
  179|  3.20M|        case '!':
  ------------------
  |  Branch (179:9): [True: 400k, False: 6.08M]
  ------------------
  180|  3.25M|        case '$': return false;
  ------------------
  |  Branch (180:9): [True: 52.0k, False: 6.43M]
  ------------------
  181|  6.48M|    }
  182|  3.22M|    return true;
  183|  6.48M|}
_ZN7jsonnet8internal20lex_get_keyword_kindERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
  207|  45.1M|{
  208|  45.1M|    auto it = keywords.find(identifier);
  209|  45.1M|    if (it == keywords.end())
  ------------------
  |  Branch (209:9): [True: 33.1M, False: 12.0M]
  ------------------
  210|  33.1M|        return Token::IDENTIFIER;
  211|  12.0M|    return it->second;
  212|  45.1M|}
_ZN7jsonnet8internal10lex_numberERPKcRKNSt3__112basic_stringIcNS4_11char_traitsIcEENS4_9allocatorIcEEEERKNS0_8LocationE:
  215|  4.04M|{
  216|       |    // This function should be understood with reference to the linked image:
  217|       |    // https://www.json.org/img/number.png
  218|       |
  219|       |    // Note, we deviate from the json.org documentation as follows:
  220|       |    // * There is no reason to lex negative numbers as atomic tokens, it is better to parse them
  221|       |    //   as a unary operator combined with a numeric literal.  This avoids x-1 being tokenized as
  222|       |    //   <identifier> <number> instead of the intended <identifier> <binop> <number>.
  223|       |    // * We support digit separators using the _ character for readability in
  224|       |    //   large numeric literals.
  225|       |
  226|  4.04M|    enum State {
  227|  4.04M|        BEGIN,
  228|  4.04M|        AFTER_ZERO,
  229|  4.04M|        AFTER_ONE_TO_NINE,
  230|  4.04M|        AFTER_INT_UNDERSCORE,
  231|  4.04M|        AFTER_DOT,
  232|  4.04M|        AFTER_DIGIT,
  233|  4.04M|        AFTER_FRAC_UNDERSCORE,
  234|  4.04M|        AFTER_E,
  235|  4.04M|        AFTER_EXP_SIGN,
  236|  4.04M|        AFTER_EXP_DIGIT,
  237|  4.04M|        AFTER_EXP_UNDERSCORE
  238|  4.04M|    } state;
  239|       |
  240|  4.04M|    std::string r;
  241|       |
  242|  4.04M|    state = BEGIN;
  243|  8.90M|    while (true) {
  ------------------
  |  Branch (243:12): [True: 8.90M, Folded]
  ------------------
  244|  8.90M|        switch (state) {
  ------------------
  |  Branch (244:17): [True: 8.90M, False: 0]
  ------------------
  245|  4.04M|            case BEGIN:
  ------------------
  |  Branch (245:13): [True: 4.04M, False: 4.85M]
  ------------------
  246|  4.04M|                switch (*c) {
  247|  1.08M|                    case '0': state = AFTER_ZERO; break;
  ------------------
  |  Branch (247:21): [True: 1.08M, False: 2.95M]
  ------------------
  248|       |
  249|  1.34M|                    case '1':
  ------------------
  |  Branch (249:21): [True: 1.34M, False: 2.70M]
  ------------------
  250|  1.62M|                    case '2':
  ------------------
  |  Branch (250:21): [True: 279k, False: 3.76M]
  ------------------
  251|  1.87M|                    case '3':
  ------------------
  |  Branch (251:21): [True: 256k, False: 3.78M]
  ------------------
  252|  2.05M|                    case '4':
  ------------------
  |  Branch (252:21): [True: 174k, False: 3.87M]
  ------------------
  253|  2.08M|                    case '5':
  ------------------
  |  Branch (253:21): [True: 30.2k, False: 4.01M]
  ------------------
  254|  2.31M|                    case '6':
  ------------------
  |  Branch (254:21): [True: 230k, False: 3.81M]
  ------------------
  255|  2.50M|                    case '7':
  ------------------
  |  Branch (255:21): [True: 193k, False: 3.85M]
  ------------------
  256|  2.74M|                    case '8':
  ------------------
  |  Branch (256:21): [True: 241k, False: 3.80M]
  ------------------
  257|  2.95M|                    case '9': state = AFTER_ONE_TO_NINE; break;
  ------------------
  |  Branch (257:21): [True: 207k, False: 3.83M]
  ------------------
  258|       |
  259|      0|                    default: throw StaticError(filename, begin, "couldn't lex number");
  ------------------
  |  Branch (259:21): [True: 0, False: 4.04M]
  ------------------
  260|  4.04M|                }
  261|  4.04M|                break;
  262|       |
  263|  4.04M|            case AFTER_ZERO:
  ------------------
  |  Branch (263:13): [True: 1.08M, False: 7.81M]
  ------------------
  264|  1.08M|                switch (*c) {
  265|  15.5k|                    case '.': state = AFTER_DOT; break;
  ------------------
  |  Branch (265:21): [True: 15.5k, False: 1.07M]
  ------------------
  266|       |
  267|    266|                    case 'e':
  ------------------
  |  Branch (267:21): [True: 266, False: 1.08M]
  ------------------
  268|    936|                    case 'E': state = AFTER_E; break;
  ------------------
  |  Branch (268:21): [True: 670, False: 1.08M]
  ------------------
  269|       |
  270|      1|                    case '_': {
  ------------------
  |  Branch (270:21): [True: 1, False: 1.08M]
  ------------------
  271|      1|                        std::stringstream ss;
  272|      1|                        ss << "couldn't lex number, _ not allowed after leading 0";
  273|      1|                        throw StaticError(filename, begin, ss.str());
  274|    266|                    }
  275|       |
  276|  1.07M|                    default: goto end;
  ------------------
  |  Branch (276:21): [True: 1.07M, False: 16.4k]
  ------------------
  277|  1.08M|                }
  278|  16.4k|                break;
  279|       |
  280|  3.54M|            case AFTER_ONE_TO_NINE:
  ------------------
  |  Branch (280:13): [True: 3.54M, False: 5.35M]
  ------------------
  281|  3.54M|                switch (*c) {
  282|  8.76k|                    case '.': state = AFTER_DOT; break;
  ------------------
  |  Branch (282:21): [True: 8.76k, False: 3.53M]
  ------------------
  283|       |
  284|    705|                    case 'e':
  ------------------
  |  Branch (284:21): [True: 705, False: 3.54M]
  ------------------
  285|  1.25k|                    case 'E': state = AFTER_E; break;
  ------------------
  |  Branch (285:21): [True: 545, False: 3.54M]
  ------------------
  286|       |
  287|   231k|                    case '0':
  ------------------
  |  Branch (287:21): [True: 231k, False: 3.31M]
  ------------------
  288|   244k|                    case '1':
  ------------------
  |  Branch (288:21): [True: 13.2k, False: 3.53M]
  ------------------
  289|   328k|                    case '2':
  ------------------
  |  Branch (289:21): [True: 83.6k, False: 3.46M]
  ------------------
  290|   358k|                    case '3':
  ------------------
  |  Branch (290:21): [True: 29.9k, False: 3.51M]
  ------------------
  291|   387k|                    case '4':
  ------------------
  |  Branch (291:21): [True: 28.8k, False: 3.51M]
  ------------------
  292|   460k|                    case '5':
  ------------------
  |  Branch (292:21): [True: 73.5k, False: 3.47M]
  ------------------
  293|   518k|                    case '6':
  ------------------
  |  Branch (293:21): [True: 57.4k, False: 3.48M]
  ------------------
  294|   541k|                    case '7':
  ------------------
  |  Branch (294:21): [True: 23.4k, False: 3.52M]
  ------------------
  295|   562k|                    case '8':
  ------------------
  |  Branch (295:21): [True: 20.7k, False: 3.52M]
  ------------------
  296|   588k|                    case '9': state = AFTER_ONE_TO_NINE; break;
  ------------------
  |  Branch (296:21): [True: 25.7k, False: 3.51M]
  ------------------
  297|       |
  298|    255|                    case '_': state = AFTER_INT_UNDERSCORE; goto skip_char;
  ------------------
  |  Branch (298:21): [True: 255, False: 3.54M]
  ------------------
  299|       |
  300|  2.94M|                    default: goto end;
  ------------------
  |  Branch (300:21): [True: 2.94M, False: 598k]
  ------------------
  301|  3.54M|                }
  302|   598k|                break;
  303|       |
  304|   598k|            case AFTER_INT_UNDERSCORE:
  ------------------
  |  Branch (304:13): [True: 255, False: 8.90M]
  ------------------
  305|    255|                switch (*c) {
  306|       |                    // The only valid transition from _ is to a digit.
  307|    188|                    case '0':
  ------------------
  |  Branch (307:21): [True: 188, False: 67]
  ------------------
  308|    233|                    case '1':
  ------------------
  |  Branch (308:21): [True: 45, False: 210]
  ------------------
  309|    238|                    case '2':
  ------------------
  |  Branch (309:21): [True: 5, False: 250]
  ------------------
  310|    238|                    case '3':
  ------------------
  |  Branch (310:21): [True: 0, False: 255]
  ------------------
  311|    244|                    case '4':
  ------------------
  |  Branch (311:21): [True: 6, False: 249]
  ------------------
  312|    244|                    case '5':
  ------------------
  |  Branch (312:21): [True: 0, False: 255]
  ------------------
  313|    244|                    case '6':
  ------------------
  |  Branch (313:21): [True: 0, False: 255]
  ------------------
  314|    244|                    case '7':
  ------------------
  |  Branch (314:21): [True: 0, False: 255]
  ------------------
  315|    244|                    case '8':
  ------------------
  |  Branch (315:21): [True: 0, False: 255]
  ------------------
  316|    244|                    case '9': state = AFTER_ONE_TO_NINE; break;
  ------------------
  |  Branch (316:21): [True: 0, False: 255]
  ------------------
  317|       |
  318|     11|                    default: {
  ------------------
  |  Branch (318:21): [True: 11, False: 244]
  ------------------
  319|     11|                        std::stringstream ss;
  320|     11|                        ss << "couldn't lex number, junk after _: " << *c;
  321|     11|                        throw StaticError(filename, begin, ss.str());
  322|    244|                    }
  323|    255|                }
  324|    244|                break;
  325|       |
  326|  24.2k|            case AFTER_DOT:
  ------------------
  |  Branch (326:13): [True: 24.2k, False: 8.88M]
  ------------------
  327|  24.2k|                switch (*c) {
  328|    578|                    case '0':
  ------------------
  |  Branch (328:21): [True: 578, False: 23.6k]
  ------------------
  329|  8.25k|                    case '1':
  ------------------
  |  Branch (329:21): [True: 7.67k, False: 16.5k]
  ------------------
  330|  8.29k|                    case '2':
  ------------------
  |  Branch (330:21): [True: 44, False: 24.2k]
  ------------------
  331|  8.69k|                    case '3':
  ------------------
  |  Branch (331:21): [True: 393, False: 23.8k]
  ------------------
  332|  8.79k|                    case '4':
  ------------------
  |  Branch (332:21): [True: 103, False: 24.1k]
  ------------------
  333|  23.9k|                    case '5':
  ------------------
  |  Branch (333:21): [True: 15.1k, False: 9.14k]
  ------------------
  334|  24.0k|                    case '6':
  ------------------
  |  Branch (334:21): [True: 172, False: 24.0k]
  ------------------
  335|  24.0k|                    case '7':
  ------------------
  |  Branch (335:21): [True: 8, False: 24.2k]
  ------------------
  336|  24.2k|                    case '8':
  ------------------
  |  Branch (336:21): [True: 144, False: 24.1k]
  ------------------
  337|  24.2k|                    case '9': state = AFTER_DIGIT; break;
  ------------------
  |  Branch (337:21): [True: 22, False: 24.2k]
  ------------------
  338|       |
  339|      9|                    default: {
  ------------------
  |  Branch (339:21): [True: 9, False: 24.2k]
  ------------------
  340|      9|                        std::stringstream ss;
  341|      9|                        ss << "couldn't lex number, junk after decimal point: " << *c;
  342|      9|                        throw StaticError(filename, begin, ss.str());
  343|  24.2k|                    }
  344|  24.2k|                }
  345|  24.2k|                break;
  346|       |
  347|   176k|            case AFTER_DIGIT:
  ------------------
  |  Branch (347:13): [True: 176k, False: 8.72M]
  ------------------
  348|   176k|                switch (*c) {
  349|     44|                    case 'e':
  ------------------
  |  Branch (349:21): [True: 44, False: 176k]
  ------------------
  350|    577|                    case 'E': state = AFTER_E; break;
  ------------------
  |  Branch (350:21): [True: 533, False: 176k]
  ------------------
  351|       |
  352|  17.0k|                    case '0':
  ------------------
  |  Branch (352:21): [True: 17.0k, False: 159k]
  ------------------
  353|  40.8k|                    case '1':
  ------------------
  |  Branch (353:21): [True: 23.7k, False: 152k]
  ------------------
  354|  48.5k|                    case '2':
  ------------------
  |  Branch (354:21): [True: 7.72k, False: 168k]
  ------------------
  355|  64.6k|                    case '3':
  ------------------
  |  Branch (355:21): [True: 16.1k, False: 160k]
  ------------------
  356|  72.6k|                    case '4':
  ------------------
  |  Branch (356:21): [True: 7.97k, False: 168k]
  ------------------
  357|  96.2k|                    case '5':
  ------------------
  |  Branch (357:21): [True: 23.5k, False: 153k]
  ------------------
  358|   112k|                    case '6':
  ------------------
  |  Branch (358:21): [True: 16.3k, False: 160k]
  ------------------
  359|   120k|                    case '7':
  ------------------
  |  Branch (359:21): [True: 7.85k, False: 168k]
  ------------------
  360|   128k|                    case '8':
  ------------------
  |  Branch (360:21): [True: 7.99k, False: 168k]
  ------------------
  361|   152k|                    case '9': state = AFTER_DIGIT; break;
  ------------------
  |  Branch (361:21): [True: 23.7k, False: 152k]
  ------------------
  362|       |
  363|    237|                    case '_': state = AFTER_FRAC_UNDERSCORE; goto skip_char;
  ------------------
  |  Branch (363:21): [True: 237, False: 176k]
  ------------------
  364|       |
  365|  23.6k|                    default: goto end;
  ------------------
  |  Branch (365:21): [True: 23.6k, False: 152k]
  ------------------
  366|   176k|                }
  367|   152k|                break;
  368|       |
  369|   152k|            case AFTER_FRAC_UNDERSCORE:
  ------------------
  |  Branch (369:13): [True: 237, False: 8.90M]
  ------------------
  370|    237|                switch (*c) {
  371|       |                    // The only valid transition from _ is to a digit.
  372|    179|                    case '0':
  ------------------
  |  Branch (372:21): [True: 179, False: 58]
  ------------------
  373|    220|                    case '1':
  ------------------
  |  Branch (373:21): [True: 41, False: 196]
  ------------------
  374|    222|                    case '2':
  ------------------
  |  Branch (374:21): [True: 2, False: 235]
  ------------------
  375|    224|                    case '3':
  ------------------
  |  Branch (375:21): [True: 2, False: 235]
  ------------------
  376|    231|                    case '4':
  ------------------
  |  Branch (376:21): [True: 7, False: 230]
  ------------------
  377|    231|                    case '5':
  ------------------
  |  Branch (377:21): [True: 0, False: 237]
  ------------------
  378|    232|                    case '6':
  ------------------
  |  Branch (378:21): [True: 1, False: 236]
  ------------------
  379|    232|                    case '7':
  ------------------
  |  Branch (379:21): [True: 0, False: 237]
  ------------------
  380|    233|                    case '8':
  ------------------
  |  Branch (380:21): [True: 1, False: 236]
  ------------------
  381|    233|                    case '9': state = AFTER_DIGIT; break;
  ------------------
  |  Branch (381:21): [True: 0, False: 237]
  ------------------
  382|       |
  383|      4|                    default: {
  ------------------
  |  Branch (383:21): [True: 4, False: 233]
  ------------------
  384|      4|                        std::stringstream ss;
  385|      4|                        ss << "couldn't lex number, junk after _: " << *c;
  386|      4|                        throw StaticError(filename, begin, ss.str());
  387|    233|                    }
  388|    237|                }
  389|    233|                break;
  390|       |
  391|  2.76k|            case AFTER_E:
  ------------------
  |  Branch (391:13): [True: 2.76k, False: 8.90M]
  ------------------
  392|  2.76k|                switch (*c) {
  393|    775|                    case '+':
  ------------------
  |  Branch (393:21): [True: 775, False: 1.98k]
  ------------------
  394|  1.36k|                    case '-': state = AFTER_EXP_SIGN; break;
  ------------------
  |  Branch (394:21): [True: 590, False: 2.17k]
  ------------------
  395|       |
  396|     74|                    case '0':
  ------------------
  |  Branch (396:21): [True: 74, False: 2.68k]
  ------------------
  397|    797|                    case '1':
  ------------------
  |  Branch (397:21): [True: 723, False: 2.04k]
  ------------------
  398|  1.06k|                    case '2':
  ------------------
  |  Branch (398:21): [True: 265, False: 2.49k]
  ------------------
  399|  1.10k|                    case '3':
  ------------------
  |  Branch (399:21): [True: 40, False: 2.72k]
  ------------------
  400|  1.14k|                    case '4':
  ------------------
  |  Branch (400:21): [True: 42, False: 2.72k]
  ------------------
  401|  1.15k|                    case '5':
  ------------------
  |  Branch (401:21): [True: 14, False: 2.74k]
  ------------------
  402|  1.17k|                    case '6':
  ------------------
  |  Branch (402:21): [True: 18, False: 2.74k]
  ------------------
  403|  1.19k|                    case '7':
  ------------------
  |  Branch (403:21): [True: 17, False: 2.74k]
  ------------------
  404|  1.27k|                    case '8':
  ------------------
  |  Branch (404:21): [True: 82, False: 2.68k]
  ------------------
  405|  1.37k|                    case '9': state = AFTER_EXP_DIGIT; break;
  ------------------
  |  Branch (405:21): [True: 97, False: 2.66k]
  ------------------
  406|       |
  407|     26|                    default: {
  ------------------
  |  Branch (407:21): [True: 26, False: 2.73k]
  ------------------
  408|     26|                        std::stringstream ss;
  409|     26|                        ss << "couldn't lex number, junk after 'E': " << *c;
  410|     26|                        throw StaticError(filename, begin, ss.str());
  411|  1.27k|                    }
  412|  2.76k|                }
  413|  2.73k|                break;
  414|       |
  415|  2.73k|            case AFTER_EXP_SIGN:
  ------------------
  |  Branch (415:13): [True: 1.36k, False: 8.90M]
  ------------------
  416|  1.36k|                switch (*c) {
  417|    941|                    case '0':
  ------------------
  |  Branch (417:21): [True: 941, False: 424]
  ------------------
  418|  1.23k|                    case '1':
  ------------------
  |  Branch (418:21): [True: 290, False: 1.07k]
  ------------------
  419|  1.23k|                    case '2':
  ------------------
  |  Branch (419:21): [True: 8, False: 1.35k]
  ------------------
  420|  1.30k|                    case '3':
  ------------------
  |  Branch (420:21): [True: 70, False: 1.29k]
  ------------------
  421|  1.32k|                    case '4':
  ------------------
  |  Branch (421:21): [True: 20, False: 1.34k]
  ------------------
  422|  1.33k|                    case '5':
  ------------------
  |  Branch (422:21): [True: 1, False: 1.36k]
  ------------------
  423|  1.33k|                    case '6':
  ------------------
  |  Branch (423:21): [True: 1, False: 1.36k]
  ------------------
  424|  1.34k|                    case '7':
  ------------------
  |  Branch (424:21): [True: 10, False: 1.35k]
  ------------------
  425|  1.34k|                    case '8':
  ------------------
  |  Branch (425:21): [True: 1, False: 1.36k]
  ------------------
  426|  1.35k|                    case '9': state = AFTER_EXP_DIGIT; break;
  ------------------
  |  Branch (426:21): [True: 17, False: 1.34k]
  ------------------
  427|       |
  428|      6|                    default: {
  ------------------
  |  Branch (428:21): [True: 6, False: 1.35k]
  ------------------
  429|      6|                        std::stringstream ss;
  430|      6|                        ss << "couldn't lex number, junk after exponent sign: " << *c;
  431|      6|                        throw StaticError(filename, begin, ss.str());
  432|  1.34k|                    }
  433|  1.36k|                }
  434|  1.35k|                break;
  435|       |
  436|  19.0k|            case AFTER_EXP_DIGIT:
  ------------------
  |  Branch (436:13): [True: 19.0k, False: 8.88M]
  ------------------
  437|  19.0k|                switch (*c) {
  438|  6.32k|                    case '0':
  ------------------
  |  Branch (438:21): [True: 6.32k, False: 12.7k]
  ------------------
  439|  8.09k|                    case '1':
  ------------------
  |  Branch (439:21): [True: 1.76k, False: 17.2k]
  ------------------
  440|  9.68k|                    case '2':
  ------------------
  |  Branch (440:21): [True: 1.59k, False: 17.4k]
  ------------------
  441|  10.4k|                    case '3':
  ------------------
  |  Branch (441:21): [True: 719, False: 18.3k]
  ------------------
  442|  11.2k|                    case '4':
  ------------------
  |  Branch (442:21): [True: 854, False: 18.1k]
  ------------------
  443|  12.1k|                    case '5':
  ------------------
  |  Branch (443:21): [True: 894, False: 18.1k]
  ------------------
  444|  12.9k|                    case '6':
  ------------------
  |  Branch (444:21): [True: 806, False: 18.2k]
  ------------------
  445|  14.4k|                    case '7':
  ------------------
  |  Branch (445:21): [True: 1.47k, False: 17.5k]
  ------------------
  446|  15.2k|                    case '8':
  ------------------
  |  Branch (446:21): [True: 822, False: 18.2k]
  ------------------
  447|  16.1k|                    case '9': state = AFTER_EXP_DIGIT; break;
  ------------------
  |  Branch (447:21): [True: 941, False: 18.1k]
  ------------------
  448|       |
  449|    128|                    case '_': state = AFTER_EXP_UNDERSCORE; goto skip_char;
  ------------------
  |  Branch (449:21): [True: 128, False: 18.9k]
  ------------------
  450|       |
  451|  2.72k|                    default: goto end;
  ------------------
  |  Branch (451:21): [True: 2.72k, False: 16.3k]
  ------------------
  452|  19.0k|                }
  453|  16.1k|                break;
  454|       |
  455|  16.1k|            case AFTER_EXP_UNDERSCORE:
  ------------------
  |  Branch (455:13): [True: 128, False: 8.90M]
  ------------------
  456|    128|                switch (*c) {
  457|       |                    // The only valid transition from _ is to a digit.
  458|     56|                    case '0':
  ------------------
  |  Branch (458:21): [True: 56, False: 72]
  ------------------
  459|     82|                    case '1':
  ------------------
  |  Branch (459:21): [True: 26, False: 102]
  ------------------
  460|     86|                    case '2':
  ------------------
  |  Branch (460:21): [True: 4, False: 124]
  ------------------
  461|     86|                    case '3':
  ------------------
  |  Branch (461:21): [True: 0, False: 128]
  ------------------
  462|     93|                    case '4':
  ------------------
  |  Branch (462:21): [True: 7, False: 121]
  ------------------
  463|     93|                    case '5':
  ------------------
  |  Branch (463:21): [True: 0, False: 128]
  ------------------
  464|    106|                    case '6':
  ------------------
  |  Branch (464:21): [True: 13, False: 115]
  ------------------
  465|    110|                    case '7':
  ------------------
  |  Branch (465:21): [True: 4, False: 124]
  ------------------
  466|    110|                    case '8':
  ------------------
  |  Branch (466:21): [True: 0, False: 128]
  ------------------
  467|    123|                    case '9': state = AFTER_EXP_DIGIT; break;
  ------------------
  |  Branch (467:21): [True: 13, False: 115]
  ------------------
  468|       |
  469|      5|                    default: {
  ------------------
  |  Branch (469:21): [True: 5, False: 123]
  ------------------
  470|      5|                        std::stringstream ss;
  471|      5|                        ss << "couldn't lex number, junk after _: " << *c;
  472|      5|                        throw StaticError(filename, begin, ss.str());
  473|    110|                    }
  474|    128|                }
  475|    123|                break;
  476|  8.90M|        }
  477|  4.85M|        r += *c;
  478|       |
  479|  4.85M|skip_char:
  480|  4.85M|        c++;
  481|  4.85M|    }
  482|  4.04M|end:
  483|  4.04M|    return r;
  484|  4.04M|}
_ZN7jsonnet8internal11jsonnet_lexERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEEPKc:
  521|  14.9k|{
  522|  14.9k|    unsigned long line_number = 1;
  523|  14.9k|    const char *line_start = input;
  524|       |
  525|  14.9k|    Tokens r;
  526|       |
  527|  14.9k|    const char *c = input;
  528|       |
  529|  14.9k|    Fodder fodder;
  530|  14.9k|    bool fresh_line = true;  // Are we tokenizing from the beginning of a new line?
  531|       |
  532|   105M|    while (*c != '\0') {
  ------------------
  |  Branch (532:12): [True: 105M, False: 6.69k]
  ------------------
  533|       |        // Used to ensure we have actually advanced the pointer by the end of the iteration.
  534|   105M|        const char *original_c = c;
  535|       |
  536|   105M|        Token::Kind kind;
  537|   105M|        std::string data;
  538|   105M|        std::string string_block_indent;
  539|   105M|        std::string string_block_term_indent;
  540|       |
  541|   105M|        unsigned new_lines, indent;
  542|   105M|        lex_ws(c, new_lines, indent, line_start, line_number);
  543|       |
  544|       |        // If it's the end of the file, discard final whitespace.
  545|   105M|        if (*c == '\0')
  ------------------
  |  Branch (545:13): [True: 7.77k, False: 105M]
  ------------------
  546|  7.77k|            break;
  547|       |
  548|   105M|        if (new_lines > 0) {
  ------------------
  |  Branch (548:13): [True: 11.2M, False: 94.5M]
  ------------------
  549|       |            // Otherwise store whitespace in fodder.
  550|  11.2M|            unsigned blanks = new_lines - 1;
  551|  11.2M|            fodder.emplace_back(FodderElement::LINE_END, blanks, indent, EMPTY);
  552|  11.2M|            fresh_line = true;
  553|  11.2M|        }
  554|       |
  555|   105M|        Location begin(line_number, c - line_start + 1);
  556|       |
  557|   105M|        switch (*c) {
  558|       |            // The following operators should never be combined with subsequent symbols.
  559|   424k|            case '{':
  ------------------
  |  Branch (559:13): [True: 424k, False: 105M]
  ------------------
  560|   424k|                kind = Token::BRACE_L;
  561|   424k|                c++;
  562|   424k|                break;
  563|       |
  564|   416k|            case '}':
  ------------------
  |  Branch (564:13): [True: 416k, False: 105M]
  ------------------
  565|   416k|                kind = Token::BRACE_R;
  566|   416k|                c++;
  567|   416k|                break;
  568|       |
  569|  2.41M|            case '[':
  ------------------
  |  Branch (569:13): [True: 2.41M, False: 103M]
  ------------------
  570|  2.41M|                kind = Token::BRACKET_L;
  571|  2.41M|                c++;
  572|  2.41M|                break;
  573|       |
  574|  2.40M|            case ']':
  ------------------
  |  Branch (574:13): [True: 2.40M, False: 103M]
  ------------------
  575|  2.40M|                kind = Token::BRACKET_R;
  576|  2.40M|                c++;
  577|  2.40M|                break;
  578|       |
  579|  8.73M|            case ',':
  ------------------
  |  Branch (579:13): [True: 8.73M, False: 97.0M]
  ------------------
  580|  8.73M|                kind = Token::COMMA;
  581|  8.73M|                c++;
  582|  8.73M|                break;
  583|       |
  584|  5.43M|            case '.':
  ------------------
  |  Branch (584:13): [True: 5.43M, False: 100M]
  ------------------
  585|  5.43M|                kind = Token::DOT;
  586|  5.43M|                c++;
  587|  5.43M|                break;
  588|       |
  589|  8.18M|            case '(':
  ------------------
  |  Branch (589:13): [True: 8.18M, False: 97.6M]
  ------------------
  590|  8.18M|                kind = Token::PAREN_L;
  591|  8.18M|                c++;
  592|  8.18M|                break;
  593|       |
  594|  8.17M|            case ')':
  ------------------
  |  Branch (594:13): [True: 8.17M, False: 97.6M]
  ------------------
  595|  8.17M|                kind = Token::PAREN_R;
  596|  8.17M|                c++;
  597|  8.17M|                break;
  598|       |
  599|  2.14M|            case ';':
  ------------------
  |  Branch (599:13): [True: 2.14M, False: 103M]
  ------------------
  600|  2.14M|                kind = Token::SEMICOLON;
  601|  2.14M|                c++;
  602|  2.14M|                break;
  603|       |
  604|       |            // Numeric literals.
  605|  1.08M|            case '0':
  ------------------
  |  Branch (605:13): [True: 1.08M, False: 104M]
  ------------------
  606|  2.43M|            case '1':
  ------------------
  |  Branch (606:13): [True: 1.34M, False: 104M]
  ------------------
  607|  2.71M|            case '2':
  ------------------
  |  Branch (607:13): [True: 279k, False: 105M]
  ------------------
  608|  2.96M|            case '3':
  ------------------
  |  Branch (608:13): [True: 256k, False: 105M]
  ------------------
  609|  3.14M|            case '4':
  ------------------
  |  Branch (609:13): [True: 174k, False: 105M]
  ------------------
  610|  3.17M|            case '5':
  ------------------
  |  Branch (610:13): [True: 30.2k, False: 105M]
  ------------------
  611|  3.40M|            case '6':
  ------------------
  |  Branch (611:13): [True: 230k, False: 105M]
  ------------------
  612|  3.59M|            case '7':
  ------------------
  |  Branch (612:13): [True: 193k, False: 105M]
  ------------------
  613|  3.83M|            case '8':
  ------------------
  |  Branch (613:13): [True: 241k, False: 105M]
  ------------------
  614|  4.04M|            case '9':
  ------------------
  |  Branch (614:13): [True: 207k, False: 105M]
  ------------------
  615|  4.04M|                kind = Token::NUMBER;
  616|  4.04M|                data = lex_number(c, filename, begin);
  617|  4.04M|                break;
  618|       |
  619|       |            // UString literals.
  620|   125k|            case '"': {
  ------------------
  |  Branch (620:13): [True: 125k, False: 105M]
  ------------------
  621|   125k|                c++;
  622|  22.9M|                for (;; ++c) {
  623|  22.9M|                    if (*c == '\0') {
  ------------------
  |  Branch (623:25): [True: 23, False: 22.9M]
  ------------------
  624|     23|                        throw StaticError(filename, begin, "unterminated string");
  625|     23|                    }
  626|  22.9M|                    if (*c == '"') {
  ------------------
  |  Branch (626:25): [True: 125k, False: 22.8M]
  ------------------
  627|   125k|                        break;
  628|   125k|                    }
  629|  22.8M|                    if (*c == '\\' && *(c + 1) != '\0') {
  ------------------
  |  Branch (629:25): [True: 84.2k, False: 22.7M]
  |  Branch (629:39): [True: 84.2k, False: 1]
  ------------------
  630|  84.2k|                        data += *c;
  631|  84.2k|                        ++c;
  632|  84.2k|                    }
  633|  22.8M|                    if (*c == '\n') {
  ------------------
  |  Branch (633:25): [True: 1.76M, False: 21.0M]
  ------------------
  634|       |                        // Maintain line/column counters.
  635|  1.76M|                        line_number++;
  636|  1.76M|                        line_start = c + 1;
  637|  1.76M|                    }
  638|  22.8M|                    data += *c;
  639|  22.8M|                }
  640|   125k|                c++;  // Advance beyond the ".
  641|   125k|                kind = Token::STRING_DOUBLE;
  642|   125k|            } break;
  643|       |
  644|       |            // UString literals.
  645|  3.55M|            case '\'': {
  ------------------
  |  Branch (645:13): [True: 3.55M, False: 102M]
  ------------------
  646|  3.55M|                c++;
  647|  47.4M|                for (;; ++c) {
  648|  47.4M|                    if (*c == '\0') {
  ------------------
  |  Branch (648:25): [True: 23, False: 47.4M]
  ------------------
  649|     23|                        throw StaticError(filename, begin, "unterminated string");
  650|     23|                    }
  651|  47.4M|                    if (*c == '\'') {
  ------------------
  |  Branch (651:25): [True: 3.55M, False: 43.9M]
  ------------------
  652|  3.55M|                        break;
  653|  3.55M|                    }
  654|  43.9M|                    if (*c == '\\' && *(c + 1) != '\0') {
  ------------------
  |  Branch (654:25): [True: 413k, False: 43.5M]
  |  Branch (654:39): [True: 413k, False: 2]
  ------------------
  655|   413k|                        data += *c;
  656|   413k|                        ++c;
  657|   413k|                    }
  658|  43.9M|                    if (*c == '\n') {
  ------------------
  |  Branch (658:25): [True: 3.02M, False: 40.9M]
  ------------------
  659|       |                        // Maintain line/column counters.
  660|  3.02M|                        line_number++;
  661|  3.02M|                        line_start = c + 1;
  662|  3.02M|                    }
  663|  43.9M|                    data += *c;
  664|  43.9M|                }
  665|  3.55M|                c++;  // Advance beyond the '.
  666|  3.55M|                kind = Token::STRING_SINGLE;
  667|  3.55M|            } break;
  668|       |
  669|       |            // Verbatim string literals.
  670|       |            // ' and " quoting is interpreted here, unlike non-verbatim strings
  671|       |            // where it is done later by jsonnet_string_unescape.  This is OK
  672|       |            // in this case because no information is lost by resoving the
  673|       |            // repeated quote into a single quote, so we can go back to the
  674|       |            // original form in the formatter.
  675|  3.46k|            case '@': {
  ------------------
  |  Branch (675:13): [True: 3.46k, False: 105M]
  ------------------
  676|  3.46k|                c++;
  677|  3.46k|                if (*c != '"' && *c != '\'') {
  ------------------
  |  Branch (677:21): [True: 1.94k, False: 1.52k]
  |  Branch (677:34): [True: 14, False: 1.92k]
  ------------------
  678|     14|                    std::stringstream ss;
  679|     14|                    ss << "couldn't lex verbatim string, junk after '@': " << *c;
  680|     14|                    throw StaticError(filename, begin, ss.str());
  681|     14|                }
  682|  3.44k|                const char quot = *c;
  683|  3.44k|                c++;  // Advance beyond the opening quote.
  684|  30.1k|                for (;; ++c) {
  685|  30.1k|                    if (*c == '\0') {
  ------------------
  |  Branch (685:25): [True: 25, False: 30.1k]
  ------------------
  686|     25|                        throw StaticError(filename, begin, "unterminated verbatim string");
  687|     25|                    }
  688|  30.1k|                    if (*c == quot) {
  ------------------
  |  Branch (688:25): [True: 4.89k, False: 25.2k]
  ------------------
  689|  4.89k|                        if (*(c + 1) == quot) {
  ------------------
  |  Branch (689:29): [True: 1.47k, False: 3.42k]
  ------------------
  690|  1.47k|                            c++;
  691|  3.42k|                        } else {
  692|  3.42k|                            break;
  693|  3.42k|                        }
  694|  4.89k|                    }
  695|  26.6k|                    data += *c;
  696|  26.6k|                }
  697|  3.42k|                c++;  // Advance beyond the closing quote.
  698|  3.42k|                if (quot == '"') {
  ------------------
  |  Branch (698:21): [True: 1.51k, False: 1.90k]
  ------------------
  699|  1.51k|                    kind = Token::VERBATIM_STRING_DOUBLE;
  700|  1.90k|                } else {
  701|  1.90k|                    kind = Token::VERBATIM_STRING_SINGLE;
  702|  1.90k|                }
  703|  3.42k|            } break;
  704|       |
  705|       |            // Keywords
  706|  59.7M|            default:
  ------------------
  |  Branch (706:13): [True: 59.7M, False: 46.0M]
  ------------------
  707|  59.7M|                if (is_identifier_first(*c)) {
  ------------------
  |  Branch (707:21): [True: 45.1M, False: 14.5M]
  ------------------
  708|  45.1M|                    std::string id;
  709|   246M|                    for (; is_identifier(*c); ++c)
  ------------------
  |  Branch (709:28): [True: 201M, False: 45.1M]
  ------------------
  710|   201M|                        id += *c;
  711|  45.1M|                    kind = lex_get_keyword_kind(id);
  712|  45.1M|                    data = id;
  713|       |
  714|  45.1M|                } else if (is_symbol(*c) || *c == '#') {
  ------------------
  |  Branch (714:28): [True: 12.4M, False: 2.10M]
  |  Branch (714:45): [True: 2.10M, False: 104]
  ------------------
  715|       |                    // Single line C++ and Python style comments.
  716|  14.5M|                    if (*c == '#' || (*c == '/' && *(c + 1) == '/')) {
  ------------------
  |  Branch (716:25): [True: 2.10M, False: 12.4M]
  |  Branch (716:39): [True: 979k, False: 11.4M]
  |  Branch (716:52): [True: 709k, False: 269k]
  ------------------
  717|  2.81M|                        std::vector<std::string> comment(1);
  718|  2.81M|                        unsigned blanks;
  719|  2.81M|                        unsigned indent;
  720|  2.81M|                        lex_until_newline(c, comment[0], blanks, indent, line_start, line_number);
  721|  2.81M|                        auto kind = fresh_line ? FodderElement::PARAGRAPH : FodderElement::LINE_END;
  ------------------
  |  Branch (721:37): [True: 2.78M, False: 37.6k]
  ------------------
  722|  2.81M|                        fodder.emplace_back(kind, blanks, indent, comment);
  723|  2.81M|                        fresh_line = true;
  724|  2.81M|                        continue;  // We've not got a token, just fodder, so keep scanning.
  725|  2.81M|                    }
  726|       |
  727|       |                    // Multi-line C style comment.
  728|  11.7M|                    if (*c == '/' && *(c + 1) == '*') {
  ------------------
  |  Branch (728:25): [True: 269k, False: 11.4M]
  |  Branch (728:38): [True: 168k, False: 100k]
  ------------------
  729|   168k|                        unsigned margin = c - line_start;
  730|       |
  731|   168k|                        const char *initial_c = c;
  732|   168k|                        c += 2;  // Avoid matching /*/: skip the /* before starting the search for
  733|       |                                 // */.
  734|       |
  735|  41.6M|                        while (!(*c == '*' && *(c + 1) == '/')) {
  ------------------
  |  Branch (735:34): [True: 544k, False: 41.0M]
  |  Branch (735:47): [True: 168k, False: 376k]
  ------------------
  736|  41.4M|                            if (*c == '\0') {
  ------------------
  |  Branch (736:33): [True: 61, False: 41.4M]
  ------------------
  737|     61|                                auto msg = "multi-line comment has no terminating */.";
  738|     61|                                throw StaticError(filename, begin, msg);
  739|     61|                            }
  740|  41.4M|                            if (*c == '\n') {
  ------------------
  |  Branch (740:33): [True: 5.53M, False: 35.9M]
  ------------------
  741|       |                                // Just keep track of the line / column counters.
  742|  5.53M|                                line_number++;
  743|  5.53M|                                line_start = c + 1;
  744|  5.53M|                            }
  745|  41.4M|                            ++c;
  746|  41.4M|                        }
  747|   168k|                        c += 2;  // Move the pointer to the char after the closing '/'.
  748|       |
  749|   168k|                        std::string comment(initial_c,
  750|   168k|                                            c - initial_c);  // Includes the "/*" and "*/".
  751|       |
  752|       |                        // Lex whitespace after comment
  753|   168k|                        unsigned new_lines_after, indent_after;
  754|   168k|                        lex_ws(c, new_lines_after, indent_after, line_start, line_number);
  755|   168k|                        std::vector<std::string> lines;
  756|   168k|                        if (comment.find('\n') >= comment.length()) {
  ------------------
  |  Branch (756:29): [True: 77.2k, False: 90.8k]
  ------------------
  757|       |                            // Comment looks like /* foo */
  758|  77.2k|                            lines.push_back(comment);
  759|  77.2k|                            fodder.emplace_back(FodderElement::INTERSTITIAL, 0, 0, lines);
  760|  77.2k|                            if (new_lines_after > 0) {
  ------------------
  |  Branch (760:33): [True: 75.4k, False: 1.79k]
  ------------------
  761|  75.4k|                                fodder.emplace_back(FodderElement::LINE_END,
  762|  75.4k|                                                    new_lines_after - 1,
  763|  75.4k|                                                    indent_after,
  764|  75.4k|                                                    EMPTY);
  765|  75.4k|                                fresh_line = true;
  766|  75.4k|                            }
  767|  90.8k|                        } else {
  768|  90.8k|                            lines = line_split(comment, margin);
  769|  90.8k|                            assert(lines[0][0] == '/');
  ------------------
  |  Branch (769:29): [True: 90.8k, False: 0]
  ------------------
  770|       |                            // Little hack to support PARAGRAPHs with * down the LHS:
  771|       |                            // Add a space to lines that start with a '*'
  772|  90.8k|                            bool all_star = true;
  773|  5.62M|                            for (auto &l : lines) {
  ------------------
  |  Branch (773:42): [True: 5.62M, False: 90.8k]
  ------------------
  774|  5.62M|                                if (l[0] != '*')
  ------------------
  |  Branch (774:37): [True: 5.55M, False: 65.7k]
  ------------------
  775|  5.55M|                                    all_star = false;
  776|  5.62M|                            }
  777|  90.8k|                            if (all_star) {
  ------------------
  |  Branch (777:33): [True: 0, False: 90.8k]
  ------------------
  778|      0|                                for (auto &l : lines) {
  ------------------
  |  Branch (778:46): [True: 0, False: 0]
  ------------------
  779|      0|                                    if (l[0] == '*')
  ------------------
  |  Branch (779:41): [True: 0, False: 0]
  ------------------
  780|      0|                                        l = " " + l;
  781|      0|                                }
  782|      0|                            }
  783|  90.8k|                            if (new_lines_after == 0) {
  ------------------
  |  Branch (783:33): [True: 4.42k, False: 86.4k]
  ------------------
  784|       |                                // Ensure a line end after the paragraph.
  785|  4.42k|                                new_lines_after = 1;
  786|  4.42k|                                indent_after = 0;
  787|  4.42k|                            }
  788|  90.8k|                            fodder_push_back(fodder,
  789|  90.8k|                                             FodderElement(FodderElement::PARAGRAPH,
  790|  90.8k|                                                           new_lines_after - 1,
  791|  90.8k|                                                           indent_after,
  792|  90.8k|                                                           lines));
  793|  90.8k|                            fresh_line = true;
  794|  90.8k|                        }
  795|   168k|                        continue;  // We've not got a token, just fodder, so keep scanning.
  796|   168k|                    }
  797|       |
  798|       |                    // Text block
  799|  11.5M|                    if (*c == '|' && *(c + 1) == '|' && *(c + 2) == '|') {
  ------------------
  |  Branch (799:25): [True: 212k, False: 11.3M]
  |  Branch (799:38): [True: 165k, False: 46.9k]
  |  Branch (799:57): [True: 6.90k, False: 159k]
  ------------------
  800|  6.90k|                        c += 3;  // Skip the "|||".
  801|       |
  802|  6.90k|                        bool chomp_trailing_nl = false;
  803|  6.90k|                        if (*c == '-') {
  ------------------
  |  Branch (803:29): [True: 904, False: 6.00k]
  ------------------
  804|    904|                            chomp_trailing_nl = true;
  805|    904|                            c++;
  806|    904|                        }
  807|       |
  808|  8.39k|                        while (is_horz_ws(*c)) ++c;  // Chomp whitespace at end of line.
  ------------------
  |  Branch (808:32): [True: 1.49k, False: 6.90k]
  ------------------
  809|  6.90k|                        if (*c != '\n') {
  ------------------
  |  Branch (809:29): [True: 42, False: 6.86k]
  ------------------
  810|     42|                            auto msg = "text block syntax requires new line after |||.";
  811|     42|                            throw StaticError(filename, begin, msg);
  812|     42|                        }
  813|  6.86k|                        std::stringstream block;
  814|  6.86k|                        c++;  // Skip the "\n"
  815|  6.86k|                        line_number++;
  816|       |                        // Skip any blank lines at the beginning of the block.
  817|  7.09k|                        while (*c == '\n') {
  ------------------
  |  Branch (817:32): [True: 228, False: 6.86k]
  ------------------
  818|    228|                            line_number++;
  819|    228|                            ++c;
  820|    228|                            block << '\n';
  821|    228|                        }
  822|  6.86k|                        line_start = c;
  823|  6.86k|                        const char *first_line = c;
  824|  6.86k|                        int ws_chars = whitespace_check(first_line, c);
  825|  6.86k|                        string_block_indent = std::string(first_line, ws_chars);
  826|  6.86k|                        if (ws_chars == 0) {
  ------------------
  |  Branch (826:29): [True: 22, False: 6.84k]
  ------------------
  827|     22|                            auto msg = "text block's first line must start with whitespace.";
  828|     22|                            throw StaticError(filename, begin, msg);
  829|     22|                        }
  830|  7.95k|                        while (true) {
  ------------------
  |  Branch (830:32): [True: 7.95k, Folded]
  ------------------
  831|  7.95k|                            assert(ws_chars > 0);
  ------------------
  |  Branch (831:29): [True: 7.95k, False: 0]
  ------------------
  832|       |                            // Read up to the \n
  833|  71.7k|                            for (c = &c[ws_chars]; *c != '\n'; ++c) {
  ------------------
  |  Branch (833:52): [True: 63.8k, False: 7.89k]
  ------------------
  834|  63.8k|                                if (*c == '\0')
  ------------------
  |  Branch (834:37): [True: 59, False: 63.8k]
  ------------------
  835|     59|                                    throw StaticError(filename, begin, "unexpected EOF");
  836|  63.8k|                                block << *c;
  837|  63.8k|                            }
  838|       |                            // Add the \n
  839|  7.89k|                            block << '\n';
  840|  7.89k|                            ++c;
  841|  7.89k|                            line_number++;
  842|  7.89k|                            line_start = c;
  843|       |                            // Skip any blank lines
  844|  8.70k|                            while (*c == '\n') {
  ------------------
  |  Branch (844:36): [True: 810, False: 7.89k]
  ------------------
  845|    810|                                line_number++;
  846|    810|                                ++c;
  847|    810|                                block << '\n';
  848|    810|                            }
  849|       |                            // Examine next line
  850|  7.89k|                            ws_chars = whitespace_check(first_line, c);
  851|  7.89k|                            if (ws_chars == 0) {
  ------------------
  |  Branch (851:33): [True: 6.78k, False: 1.11k]
  ------------------
  852|       |                                // End of text block (or indentation error).
  853|       |                                // Count actual whitespace on this line.
  854|  6.78k|                                int actual_ws = 0;
  855|  51.9k|                                while (c[actual_ws] == ' ' ||
  ------------------
  |  Branch (855:40): [True: 24.5k, False: 27.4k]
  ------------------
  856|  45.1k|                                       c[actual_ws] == '\t') {
  ------------------
  |  Branch (856:40): [True: 20.6k, False: 6.78k]
  ------------------
  857|  45.1k|                                    actual_ws++;
  858|  45.1k|                                }
  859|       |
  860|       |                                // Check if this is the terminator |||
  861|  6.78k|                                bool is_terminator = (
  862|  6.78k|                                    c[actual_ws] == '|' &&
  ------------------
  |  Branch (862:37): [True: 6.71k, False: 69]
  ------------------
  863|  6.71k|                                    c[actual_ws + 1] == '|' &&
  ------------------
  |  Branch (863:37): [True: 6.70k, False: 7]
  ------------------
  864|  6.70k|                                    c[actual_ws + 2] == '|');
  ------------------
  |  Branch (864:37): [True: 6.69k, False: 10]
  ------------------
  865|       |
  866|  6.78k|                                if (!is_terminator) {
  ------------------
  |  Branch (866:37): [True: 86, False: 6.69k]
  ------------------
  867|       |                                    // Not a terminator - check if it's an
  868|       |                                    // indentation issue.
  869|     86|                                    if (actual_ws > 0) {
  ------------------
  |  Branch (869:41): [True: 44, False: 42]
  ------------------
  870|       |                                        // Has whitespace but doesn't match expected
  871|       |                                        // indentation.
  872|     44|                                        std::stringstream msg;
  873|     44|                                        msg << "text block indentation mismatch: "
  874|     44|                                                "expected at least ";
  875|     44|                                        describe_whitespace(msg, string_block_indent);
  876|     44|                                        msg << ", found ";
  877|     44|                                        describe_whitespace(msg, std::string(c, actual_ws));
  878|     44|                                        throw StaticError(filename, begin, msg.str());
  879|     44|                                    } else {
  880|       |                                        // No whitespace and no ||| - missing
  881|       |                                        // terminator.
  882|     42|                                        auto msg =
  883|     42|                                            "text block not terminated with |||";
  884|     42|                                        throw StaticError(filename, begin, msg);
  885|     42|                                    }
  886|     86|                                }
  887|       |
  888|       |                                // Valid termination - skip over any whitespace.
  889|  48.0k|                                while (*c == ' ' || *c == '\t') {
  ------------------
  |  Branch (889:40): [True: 23.9k, False: 24.1k]
  |  Branch (889:53): [True: 17.4k, False: 6.69k]
  ------------------
  890|  41.4k|                                    string_block_term_indent += *c;
  891|  41.4k|                                    ++c;
  892|  41.4k|                                }
  893|       |                                // Skip the |||
  894|  6.69k|                                c += 3;  // Leave after the last |
  895|  6.69k|                                data = block.str();
  896|  6.69k|                                kind = Token::STRING_BLOCK;
  897|  6.69k|                                if (chomp_trailing_nl) {
  ------------------
  |  Branch (897:37): [True: 890, False: 5.80k]
  ------------------
  898|    890|                                    assert(data.back() == '\n');
  ------------------
  |  Branch (898:37): [True: 890, False: 0]
  ------------------
  899|    890|                                    data.pop_back();
  900|    890|                                }
  901|  6.69k|                                break;  // Out of the while loop.
  902|  6.69k|                            }
  903|  7.89k|                        }
  904|       |
  905|  6.69k|                        break;  // Out of the switch.
  906|  6.84k|                    }
  907|       |
  908|  11.5M|                    const char *operator_begin = c;
  909|  40.5M|                    for (; is_symbol(*c); ++c) {
  ------------------
  |  Branch (909:28): [True: 28.9M, False: 11.5M]
  ------------------
  910|       |                        // Not allowed // in operators
  911|  28.9M|                        if (*c == '/' && *(c + 1) == '/')
  ------------------
  |  Branch (911:29): [True: 101k, False: 28.8M]
  |  Branch (911:42): [True: 407, False: 101k]
  ------------------
  912|    407|                            break;
  913|       |                        // Not allowed /* in operators
  914|  28.9M|                        if (*c == '/' && *(c + 1) == '*')
  ------------------
  |  Branch (914:29): [True: 101k, False: 28.8M]
  |  Branch (914:42): [True: 452, False: 101k]
  ------------------
  915|    452|                            break;
  916|       |                        // Not allowed ||| in operators
  917|  28.9M|                        if (*c == '|' && *(c + 1) == '|' && *(c + 2) == '|')
  ------------------
  |  Branch (917:29): [True: 368k, False: 28.6M]
  |  Branch (917:42): [True: 162k, False: 206k]
  |  Branch (917:61): [True: 2.61k, False: 159k]
  ------------------
  918|  2.61k|                            break;
  919|  28.9M|                    }
  920|       |                    // Not allowed to end with a + - ~ ! unless a single char.
  921|       |                    // So, wind it back if we need to (but not too far).
  922|  14.8M|                    while (c > operator_begin + 1 && !allowed_at_end_of_operator(*(c - 1))) {
  ------------------
  |  Branch (922:28): [True: 6.48M, False: 8.34M]
  |  Branch (922:54): [True: 3.25M, False: 3.22M]
  ------------------
  923|  3.25M|                        c--;
  924|  3.25M|                    }
  925|  11.5M|                    data += std::string(operator_begin, c);
  926|  11.5M|                    if (data == "$") {
  ------------------
  |  Branch (926:25): [True: 26.4k, False: 11.5M]
  ------------------
  927|  26.4k|                        kind = Token::DOLLAR;
  928|  26.4k|                        data = "";
  929|  11.5M|                    } else {
  930|  11.5M|                        kind = Token::OPERATOR;
  931|  11.5M|                    }
  932|  11.5M|                } else {
  933|    104|                    std::stringstream ss;
  934|    104|                    ss << "Could not lex the character ";
  935|    104|                    auto uc = (unsigned char)(*c);
  936|    104|                    if (*c < 32)
  ------------------
  |  Branch (936:25): [True: 94, False: 10]
  ------------------
  937|     94|                        ss << "code " << unsigned(uc);
  938|     10|                    else
  939|     10|                        ss << "'" << *c << "'";
  940|    104|                    throw StaticError(filename, begin, ss.str());
  941|    104|                }
  942|   105M|        }
  943|       |
  944|       |        // Ensure that a bug in the above code does not cause an infinite memory consuming loop due
  945|       |        // to pushing empty tokens.
  946|   102M|        if (c == original_c) {
  ------------------
  |  Branch (946:13): [True: 0, False: 102M]
  ------------------
  947|      0|            throw StaticError(filename, begin, "internal lexing error:  pointer did not advance");
  948|      0|        }
  949|       |
  950|   102M|        Location end(line_number, (c + 1) - line_start);
  951|   102M|        r.emplace_back(kind,
  952|   102M|                       fodder,
  953|   102M|                       data,
  954|   102M|                       string_block_indent,
  955|   102M|                       string_block_term_indent,
  956|   102M|                       LocationRange(filename, begin, end));
  957|   102M|        fodder.clear();
  958|   102M|        fresh_line = false;
  959|   102M|    }
  960|       |
  961|  14.4k|    Location begin(line_number, c - line_start + 1);
  962|  14.4k|    Location end(line_number, (c + 1) - line_start + 1);
  963|  14.4k|    r.emplace_back(Token::END_OF_FILE, fodder, "", "", "", LocationRange(filename, begin, end));
  964|  14.4k|    return r;
  965|  14.9k|}
lexer.cpp:_ZN7jsonnet8internalL6lex_wsERPKcRjS4_S3_Rm:
   81|   108M|{
   82|   108M|    indent = 0;
   83|   108M|    new_lines = 0;
   84|   253M|    for (; *c != '\0' && is_ws(*c); c++) {
  ------------------
  |  Branch (84:12): [True: 253M, False: 8.31k]
  |  Branch (84:26): [True: 144M, False: 108M]
  ------------------
   85|   144M|        switch (*c) {
  ------------------
  |  Branch (85:17): [True: 144M, False: 0]
  ------------------
   86|  88.8k|            case '\r':
  ------------------
  |  Branch (86:13): [True: 88.8k, False: 144M]
  ------------------
   87|       |                // Ignore.
   88|  88.8k|                break;
   89|       |
   90|  15.7M|            case '\n':
  ------------------
  |  Branch (90:13): [True: 15.7M, False: 128M]
  ------------------
   91|  15.7M|                indent = 0;
   92|  15.7M|                new_lines++;
   93|  15.7M|                line_number++;
   94|  15.7M|                line_start = c + 1;
   95|  15.7M|                break;
   96|       |
   97|   128M|            case ' ': indent += 1; break;
  ------------------
  |  Branch (97:13): [True: 128M, False: 15.8M]
  ------------------
   98|       |
   99|       |            // This only works for \t at the beginning of lines, but we strip it everywhere else
  100|       |            // anyway.  The only case where this will cause a problem is spaces followed by \t
  101|       |            // at the beginning of a line.  However that is rare, ill-advised, and if re-indentation
  102|       |            // is enabled it will be fixed later.
  103|  12.0k|            case '\t': indent += 8; break;
  ------------------
  |  Branch (103:13): [True: 12.0k, False: 144M]
  ------------------
  104|   144M|        }
  105|   144M|    }
  106|   108M|}
lexer.cpp:_ZN7jsonnet8internalL5is_wsEc:
   39|   253M|{
   40|   253M|    return c == '\n' || is_horz_ws(c);
  ------------------
  |  Branch (40:12): [True: 15.7M, False: 237M]
  |  Branch (40:25): [True: 128M, False: 108M]
  ------------------
   41|   253M|}
lexer.cpp:_ZN7jsonnet8internalL19is_identifier_firstEc:
  143|   306M|{
  144|   306M|    return is_upper(c) || is_lower(c) || c == '_';
  ------------------
  |  Branch (144:12): [True: 5.75M, False: 300M]
  |  Branch (144:27): [True: 226M, False: 73.9M]
  |  Branch (144:42): [True: 3.16M, False: 70.8M]
  ------------------
  145|   306M|}
lexer.cpp:_ZN7jsonnet8internalL8is_upperEc:
  128|   306M|{
  129|   306M|    return c >= 'A' && c <= 'Z';
  ------------------
  |  Branch (129:12): [True: 238M, False: 68.0M]
  |  Branch (129:24): [True: 5.75M, False: 232M]
  ------------------
  130|   306M|}
lexer.cpp:_ZN7jsonnet8internalL8is_lowerEc:
  133|   300M|{
  134|   300M|    return c >= 'a' && c <= 'z';
  ------------------
  |  Branch (134:12): [True: 227M, False: 73.7M]
  |  Branch (134:24): [True: 226M, False: 222k]
  ------------------
  135|   300M|}
lexer.cpp:_ZN7jsonnet8internalL13is_identifierEc:
  148|   246M|{
  149|   246M|    return is_identifier_first(c) || is_number(c);
  ------------------
  |  Branch (149:12): [True: 190M, False: 56.2M]
  |  Branch (149:38): [True: 11.0M, False: 45.1M]
  ------------------
  150|   246M|}
lexer.cpp:_ZN7jsonnet8internalL9is_numberEc:
  138|  56.2M|{
  139|  56.2M|    return c >= '0' && c <= '9';
  ------------------
  |  Branch (139:12): [True: 15.0M, False: 41.2M]
  |  Branch (139:24): [True: 11.0M, False: 3.95M]
  ------------------
  140|  56.2M|}
lexer.cpp:_ZN7jsonnet8internalL9is_symbolEc:
  153|  55.1M|{
  154|  55.1M|    switch (c) {
  ------------------
  |  Branch (154:13): [True: 41.4M, False: 13.6M]
  ------------------
  155|  1.30M|        case '!':
  ------------------
  |  Branch (155:9): [True: 1.30M, False: 53.8M]
  ------------------
  156|  1.40M|        case '$':
  ------------------
  |  Branch (156:9): [True: 105k, False: 55.0M]
  ------------------
  157|  6.93M|        case ':':
  ------------------
  |  Branch (157:9): [True: 5.52M, False: 49.5M]
  ------------------
  158|  7.40M|        case '~':
  ------------------
  |  Branch (158:9): [True: 468k, False: 54.6M]
  ------------------
  159|  15.7M|        case '+':
  ------------------
  |  Branch (159:9): [True: 8.31M, False: 46.7M]
  ------------------
  160|  16.5M|        case '-':
  ------------------
  |  Branch (160:9): [True: 856k, False: 54.2M]
  ------------------
  161|  17.5M|        case '&':
  ------------------
  |  Branch (161:9): [True: 996k, False: 54.1M]
  ------------------
  162|  18.1M|        case '|':
  ------------------
  |  Branch (162:9): [True: 581k, False: 54.5M]
  ------------------
  163|  18.1M|        case '^':
  ------------------
  |  Branch (163:9): [True: 6.64k, False: 55.1M]
  ------------------
  164|  26.7M|        case '=':
  ------------------
  |  Branch (164:9): [True: 8.60M, False: 46.5M]
  ------------------
  165|  27.7M|        case '<':
  ------------------
  |  Branch (165:9): [True: 943k, False: 54.1M]
  ------------------
  166|  28.9M|        case '>':
  ------------------
  |  Branch (166:9): [True: 1.24M, False: 53.8M]
  ------------------
  167|  39.8M|        case '*':
  ------------------
  |  Branch (167:9): [True: 10.9M, False: 44.1M]
  ------------------
  168|  40.9M|        case '/':
  ------------------
  |  Branch (168:9): [True: 1.08M, False: 54.0M]
  ------------------
  169|  41.4M|        case '%': return true;
  ------------------
  |  Branch (169:9): [True: 482k, False: 54.6M]
  ------------------
  170|  55.1M|    }
  171|  13.6M|    return false;
  172|  55.1M|}
lexer.cpp:_ZN7jsonnet8internalL17lex_until_newlineERPKcRNSt3__112basic_stringIcNS4_11char_traitsIcEENS4_9allocatorIcEEEERjSC_S3_Rm:
  113|  2.81M|{
  114|  2.81M|    const char *original_c = c;
  115|  2.81M|    const char *last_non_space = c;
  116|  42.5M|    for (; *c != '\0' && *c != '\n'; c++) {
  ------------------
  |  Branch (116:12): [True: 42.5M, False: 251]
  |  Branch (116:26): [True: 39.7M, False: 2.81M]
  ------------------
  117|  39.7M|        if (!is_horz_ws(*c))
  ------------------
  |  Branch (117:13): [True: 34.0M, False: 5.66M]
  ------------------
  118|  34.0M|            last_non_space = c;
  119|  39.7M|    }
  120|  2.81M|    text = std::string(original_c, last_non_space - original_c + 1);
  121|       |    // Consume subsequent whitespace including the '\n'.
  122|  2.81M|    unsigned new_lines;
  123|  2.81M|    lex_ws(c, new_lines, indent, line_start, line_number);
  124|  2.81M|    blanks = new_lines == 0 ? 0 : new_lines - 1;
  ------------------
  |  Branch (124:14): [True: 251, False: 2.81M]
  ------------------
  125|  2.81M|}
lexer.cpp:_ZN7jsonnet8internalL10line_splitERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEEj:
   60|  90.8k|{
   61|  90.8k|    std::vector<std::string> ret;
   62|  90.8k|    std::stringstream ss;
   63|  39.6M|    for (size_t i = 0; i < s.length(); ++i) {
  ------------------
  |  Branch (63:24): [True: 39.5M, False: 90.8k]
  ------------------
   64|  39.5M|        if (s[i] == '\n') {
  ------------------
  |  Branch (64:13): [True: 5.53M, False: 33.9M]
  ------------------
   65|  5.53M|            ret.emplace_back(strip_ws(ss.str(), margin));
   66|  5.53M|            ss.str("");
   67|  33.9M|        } else {
   68|  33.9M|            ss << s[i];
   69|  33.9M|        }
   70|  39.5M|    }
   71|  90.8k|    ret.emplace_back(strip_ws(ss.str(), margin));
   72|  90.8k|    return ret;
   73|  90.8k|}
lexer.cpp:_ZN7jsonnet8internalL8strip_wsERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEEj:
   45|  5.62M|{
   46|  5.62M|    if (s.size() == 0)
  ------------------
  |  Branch (46:9): [True: 4.62M, False: 998k]
  ------------------
   47|  4.62M|        return s;  // Avoid underflow below.
   48|   998k|    size_t i = 0;
   49|  2.82M|    while (i < s.length() && is_horz_ws(s[i]) && i < margin)
  ------------------
  |  Branch (49:12): [True: 2.82M, False: 1.30k]
  |  Branch (49:30): [True: 2.21M, False: 610k]
  |  Branch (49:50): [True: 1.82M, False: 386k]
  ------------------
   50|  1.82M|        i++;
   51|   998k|    size_t j = s.size();
   52|  2.51M|    while (j > i && is_horz_ws(s[j - 1])) {
  ------------------
  |  Branch (52:12): [True: 2.42M, False: 94.2k]
  |  Branch (52:21): [True: 1.51M, False: 903k]
  ------------------
   53|  1.51M|        j--;
   54|  1.51M|    }
   55|   998k|    return std::string(&s[i], &s[j]);
   56|  5.62M|}
lexer.cpp:_ZN7jsonnet8internalL10is_horz_wsEc:
   33|   282M|{
   34|   282M|    return c == ' ' || c == '\t' || c == '\r';
  ------------------
  |  Branch (34:12): [True: 138M, False: 144M]
  |  Branch (34:24): [True: 25.9k, False: 144M]
  |  Branch (34:37): [True: 100k, False: 144M]
  ------------------
   35|   282M|}
lexer.cpp:_ZN7jsonnet8internalL16whitespace_checkEPKcS2_:
  489|  14.7k|{
  490|  14.7k|    int i = 0;
  491|   182k|    while (a[i] == ' ' || a[i] == '\t') {
  ------------------
  |  Branch (491:12): [True: 15.4k, False: 167k]
  |  Branch (491:27): [True: 159k, False: 7.98k]
  ------------------
  492|   174k|        if (b[i] != a[i])
  ------------------
  |  Branch (492:13): [True: 6.78k, False: 167k]
  ------------------
  493|  6.78k|            return 0;
  494|   167k|        i++;
  495|   167k|    }
  496|  7.98k|    return i;
  497|  14.7k|}
lexer.cpp:_ZN7jsonnet8internalL19describe_whitespaceERNSt3__118basic_stringstreamIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERKNS1_12basic_stringIcS4_S6_EE:
  499|     88|static void describe_whitespace(std::stringstream& msg, const std::string& ws) {
  500|     88|    int spaces = 0;
  501|     88|    int tabs = 0;
  502|  22.4k|    for (char c : ws) {
  ------------------
  |  Branch (502:17): [True: 22.4k, False: 88]
  ------------------
  503|  22.4k|        if (c == ' ')
  ------------------
  |  Branch (503:13): [True: 680, False: 21.7k]
  ------------------
  504|    680|            spaces++;
  505|  21.7k|        else if (c == '\t')
  ------------------
  |  Branch (505:18): [True: 21.7k, False: 0]
  ------------------
  506|  21.7k|            tabs++;
  507|  22.4k|    }
  508|     88|    if (spaces > 0 && tabs > 0) {
  ------------------
  |  Branch (508:9): [True: 42, False: 46]
  |  Branch (508:23): [True: 14, False: 28]
  ------------------
  509|     14|        msg << spaces << (spaces == 1 ? " space" : " spaces") << " and " << tabs
  ------------------
  |  Branch (509:27): [True: 5, False: 9]
  ------------------
  510|     14|            << (tabs == 1 ? " tab" : " tabs");
  ------------------
  |  Branch (510:17): [True: 5, False: 9]
  ------------------
  511|     74|    } else if (spaces > 0) {
  ------------------
  |  Branch (511:16): [True: 28, False: 46]
  ------------------
  512|     28|        msg << spaces << (spaces == 1 ? " space" : " spaces");
  ------------------
  |  Branch (512:27): [True: 9, False: 19]
  ------------------
  513|     46|    } else if (tabs > 0) {
  ------------------
  |  Branch (513:16): [True: 46, False: 0]
  ------------------
  514|     46|        msg << tabs << (tabs == 1 ? " tab" : " tabs");
  ------------------
  |  Branch (514:25): [True: 19, False: 27]
  ------------------
  515|     46|    } else {
  516|      0|        msg << "no indentation";
  517|      0|    }
  518|     88|}

_ZN7jsonnet8internal13FodderElementC2ENS1_4KindEjjRKNSt3__16vectorINS3_12basic_stringIcNS3_11char_traitsIcEENS3_9allocatorIcEEEENS8_ISA_EEEE:
   92|  14.3M|        : kind(kind), blanks(blanks), indent(indent), comment(comment)
   93|  14.3M|    {
   94|  14.3M|        assert(kind != LINE_END || comment.size() <= 1);
  ------------------
  |  Branch (94:9): [True: 2.95M, False: 11.4M]
  |  Branch (94:9): [True: 11.4M, False: 0]
  |  Branch (94:9): [True: 14.3M, False: 0]
  ------------------
   95|  14.3M|        assert(kind != INTERSTITIAL || (blanks == 0 && indent == 0 && comment.size() == 1));
  ------------------
  |  Branch (95:9): [True: 77.2k, False: 0]
  |  Branch (95:9): [True: 77.2k, False: 0]
  |  Branch (95:9): [True: 77.2k, False: 0]
  |  Branch (95:9): [True: 14.2M, False: 77.2k]
  |  Branch (95:9): [True: 14.3M, False: 0]
  ------------------
   96|  14.3M|        assert(kind != PARAGRAPH || comment.size() >= 1);
  ------------------
  |  Branch (96:9): [True: 11.4M, False: 2.87M]
  |  Branch (96:9): [True: 2.87M, False: 0]
  |  Branch (96:9): [True: 14.3M, False: 0]
  ------------------
   97|  14.3M|    }
_ZNK7jsonnet8internal5Token6data32Ev:
  299|  36.7M|    {
  300|  36.7M|        return decode_utf8(data);
  301|  36.7M|    }
_ZN7jsonnet8internal5TokenC2ENS1_4KindERKNSt3__16vectorINS0_13FodderElementENS3_9allocatorIS5_EEEERKNS3_12basic_stringIcNS3_11char_traitsIcEENS6_IcEEEESH_SH_RKNS0_13LocationRangeE:
  308|   102M|        : kind(kind),
  309|   102M|          fodder(fodder),
  310|   102M|          data(data),
  311|   102M|          stringBlockIndent(string_block_indent),
  312|   102M|          stringBlockTermIndent(string_block_term_indent),
  313|   102M|          location(location)
  314|   102M|    {
  315|   102M|    }
_ZN7jsonnet8internal5Token8toStringENS1_4KindE:
  320|  1.00k|    {
  321|  1.00k|        switch (v) {
  322|     20|            case BRACE_L: return "\"{\"";
  ------------------
  |  Branch (322:13): [True: 20, False: 981]
  ------------------
  323|     24|            case BRACE_R: return "\"}\"";
  ------------------
  |  Branch (323:13): [True: 24, False: 977]
  ------------------
  324|      2|            case BRACKET_L: return "\"[\"";
  ------------------
  |  Branch (324:13): [True: 2, False: 999]
  ------------------
  325|     65|            case BRACKET_R: return "\"]\"";
  ------------------
  |  Branch (325:13): [True: 65, False: 936]
  ------------------
  326|     45|            case COMMA: return "\",\"";
  ------------------
  |  Branch (326:13): [True: 45, False: 956]
  ------------------
  327|     16|            case DOLLAR: return "\"$\"";
  ------------------
  |  Branch (327:13): [True: 16, False: 985]
  ------------------
  328|     12|            case DOT: return "\".\"";
  ------------------
  |  Branch (328:13): [True: 12, False: 989]
  ------------------
  329|       |
  330|      2|            case PAREN_L: return "\"(\"";
  ------------------
  |  Branch (330:13): [True: 2, False: 999]
  ------------------
  331|     29|            case PAREN_R: return "\")\"";
  ------------------
  |  Branch (331:13): [True: 29, False: 972]
  ------------------
  332|     24|            case SEMICOLON: return "\";\"";
  ------------------
  |  Branch (332:13): [True: 24, False: 977]
  ------------------
  333|       |
  334|    131|            case IDENTIFIER: return "IDENTIFIER";
  ------------------
  |  Branch (334:13): [True: 131, False: 870]
  ------------------
  335|     58|            case NUMBER: return "NUMBER";
  ------------------
  |  Branch (335:13): [True: 58, False: 943]
  ------------------
  336|    148|            case OPERATOR: return "OPERATOR";
  ------------------
  |  Branch (336:13): [True: 148, False: 853]
  ------------------
  337|     14|            case STRING_SINGLE: return "STRING_SINGLE";
  ------------------
  |  Branch (337:13): [True: 14, False: 987]
  ------------------
  338|     11|            case STRING_DOUBLE: return "STRING_DOUBLE";
  ------------------
  |  Branch (338:13): [True: 11, False: 990]
  ------------------
  339|      6|            case VERBATIM_STRING_SINGLE: return "VERBATIM_STRING_SINGLE";
  ------------------
  |  Branch (339:13): [True: 6, False: 995]
  ------------------
  340|      2|            case VERBATIM_STRING_DOUBLE: return "VERBATIM_STRING_DOUBLE";
  ------------------
  |  Branch (340:13): [True: 2, False: 999]
  ------------------
  341|      5|            case STRING_BLOCK: return "STRING_BLOCK";
  ------------------
  |  Branch (341:13): [True: 5, False: 996]
  ------------------
  342|       |
  343|      1|            case ASSERT: return "assert";
  ------------------
  |  Branch (343:13): [True: 1, False: 1.00k]
  ------------------
  344|      1|            case ELSE: return "else";
  ------------------
  |  Branch (344:13): [True: 1, False: 1.00k]
  ------------------
  345|      0|            case ERROR: return "error";
  ------------------
  |  Branch (345:13): [True: 0, False: 1.00k]
  ------------------
  346|      0|            case FALSE: return "false";
  ------------------
  |  Branch (346:13): [True: 0, False: 1.00k]
  ------------------
  347|      1|            case FOR: return "for";
  ------------------
  |  Branch (347:13): [True: 1, False: 1.00k]
  ------------------
  348|      3|            case FUNCTION: return "function";
  ------------------
  |  Branch (348:13): [True: 3, False: 998]
  ------------------
  349|      1|            case IF: return "if";
  ------------------
  |  Branch (349:13): [True: 1, False: 1.00k]
  ------------------
  350|      0|            case IMPORT: return "import";
  ------------------
  |  Branch (350:13): [True: 0, False: 1.00k]
  ------------------
  351|      1|            case IMPORTSTR: return "importstr";
  ------------------
  |  Branch (351:13): [True: 1, False: 1.00k]
  ------------------
  352|      0|            case IMPORTBIN: return "importbin";
  ------------------
  |  Branch (352:13): [True: 0, False: 1.00k]
  ------------------
  353|     13|            case IN: return "in";
  ------------------
  |  Branch (353:13): [True: 13, False: 988]
  ------------------
  354|      0|            case LOCAL: return "local";
  ------------------
  |  Branch (354:13): [True: 0, False: 1.00k]
  ------------------
  355|      0|            case NULL_LIT: return "null";
  ------------------
  |  Branch (355:13): [True: 0, False: 1.00k]
  ------------------
  356|      0|            case SELF: return "self";
  ------------------
  |  Branch (356:13): [True: 0, False: 1.00k]
  ------------------
  357|      0|            case SUPER: return "super";
  ------------------
  |  Branch (357:13): [True: 0, False: 1.00k]
  ------------------
  358|      2|            case TAILSTRICT: return "tailstrict";
  ------------------
  |  Branch (358:13): [True: 2, False: 999]
  ------------------
  359|     10|            case THEN: return "then";
  ------------------
  |  Branch (359:13): [True: 10, False: 991]
  ------------------
  360|      0|            case TRUE: return "true";
  ------------------
  |  Branch (360:13): [True: 0, False: 1.00k]
  ------------------
  361|       |
  362|    354|            case END_OF_FILE: return "end of file";
  ------------------
  |  Branch (362:13): [True: 354, False: 647]
  ------------------
  363|      0|            default:
  ------------------
  |  Branch (363:13): [True: 0, False: 1.00k]
  ------------------
  364|      0|                std::cerr << "INTERNAL ERROR: Unknown token kind: " << v << std::endl;
  365|      0|                std::abort();
  366|  1.00k|        }
  367|  1.00k|    }
parser.cpp:_ZN7jsonnet8internallsERNSt3__113basic_ostreamIcNS1_11char_traitsIcEEEENS0_5Token4KindE:
  387|    576|{
  388|    576|    o << Token::toString(v);
  389|    576|    return o;
  390|    576|}
parser.cpp:_ZN7jsonnet8internallsERNSt3__113basic_ostreamIcNS1_11char_traitsIcEEEERKNS0_5TokenE:
  393|    460|{
  394|    460|    if (v.data == "") {
  ------------------
  |  Branch (394:9): [True: 302, False: 158]
  ------------------
  395|    302|        o << Token::toString(v.kind);
  396|    302|    } else if (v.kind == Token::OPERATOR) {
  ------------------
  |  Branch (396:16): [True: 35, False: 123]
  ------------------
  397|     35|        o << "\"" << v.data << "\"";
  398|    123|    } else {
  399|    123|        o << "(" << Token::toString(v.kind) << ", \"" << v.data << "\")";
  400|    123|    }
  401|    460|    return o;
  402|    460|}
lexer.cpp:_ZN7jsonnet8internalL16fodder_push_backERNSt3__16vectorINS0_13FodderElementENS1_9allocatorIS3_EEEERKS3_:
  143|  90.8k|{
  144|  90.8k|    if (fodder_has_clean_endline(a) && elem.kind == FodderElement::LINE_END) {
  ------------------
  |  Branch (144:9): [True: 81.1k, False: 9.72k]
  |  Branch (144:40): [True: 0, False: 81.1k]
  ------------------
  145|      0|        if (elem.comment.size() > 0) {
  ------------------
  |  Branch (145:13): [True: 0, False: 0]
  ------------------
  146|       |            // The line end had a comment, so create a single line paragraph for it.
  147|      0|            a.emplace_back(FodderElement::PARAGRAPH, elem.blanks, elem.indent, elem.comment);
  148|      0|        } else {
  149|       |            // Merge it into the previous line end.
  150|      0|            a.back().indent = elem.indent;
  151|      0|            a.back().blanks += elem.blanks;
  152|      0|        }
  153|  90.8k|    } else {
  154|  90.8k|        if (!fodder_has_clean_endline(a) && elem.kind == FodderElement::PARAGRAPH) {
  ------------------
  |  Branch (154:13): [True: 9.72k, False: 81.1k]
  |  Branch (154:45): [True: 9.72k, False: 0]
  ------------------
  155|  9.72k|            a.emplace_back(FodderElement::LINE_END, 0, elem.indent, std::vector<std::string>());
  156|  9.72k|        }
  157|  90.8k|        a.push_back(elem);
  158|  90.8k|    }
  159|  90.8k|}
lexer.cpp:_ZN7jsonnet8internalL24fodder_has_clean_endlineERKNSt3__16vectorINS0_13FodderElementENS1_9allocatorIS3_EEEE:
  134|   181k|{
  135|   181k|    return !fodder.empty() && fodder.back().kind != FodderElement::INTERSTITIAL;
  ------------------
  |  Branch (135:12): [True: 163k, False: 17.9k]
  |  Branch (135:31): [True: 162k, False: 1.47k]
  ------------------
  136|   181k|}

jsonnet_version:
  334|  14.8k|{
  335|  14.8k|    return LIB_JSONNET_VERSION;
  ------------------
  |  |   34|  14.8k|#define LIB_JSONNET_VERSION "v0.22.0"
  ------------------
  336|  14.8k|}
jsonnet_make:
  339|  7.42k|{
  340|  7.42k|    TRY
  ------------------
  |  |  319|  7.42k|#define TRY try {
  ------------------
  341|  7.42k|        return new JsonnetVm();
  342|  7.42k|    CATCH("jsonnet_make")
  ------------------
  |  |  321|  7.42k|    }                                                                                         \
  |  |  322|  7.42k|    catch (const std::bad_alloc &)                                                            \
  |  |  323|  7.42k|    {                                                                                         \
  |  |  324|      0|        memory_panic();                                                                       \
  |  |  325|      0|    }                                                                                         \
  |  |  326|  7.42k|    catch (const std::exception &e)                                                           \
  |  |  327|  7.42k|    {                                                                                         \
  |  |  328|      0|        std::cerr << "Something went wrong during " func ", please report this: " << e.what() \
  |  |  329|      0|                  << std::endl;                                                               \
  |  |  330|      0|        abort();                                                                              \
  |  |  331|      0|    }
  ------------------
  343|      0|    return nullptr;
  344|  7.42k|}
jsonnet_destroy:
  347|  7.42k|{
  348|  7.42k|    TRY
  ------------------
  |  |  319|  7.42k|#define TRY try {
  ------------------
  349|  7.42k|        delete vm;
  350|  7.42k|    CATCH("jsonnet_destroy")
  ------------------
  |  |  321|  7.42k|    }                                                                                         \
  |  |  322|  7.42k|    catch (const std::bad_alloc &)                                                            \
  |  |  323|  7.42k|    {                                                                                         \
  |  |  324|      0|        memory_panic();                                                                       \
  |  |  325|      0|    }                                                                                         \
  |  |  326|  7.42k|    catch (const std::exception &e)                                                           \
  |  |  327|  7.42k|    {                                                                                         \
  |  |  328|      0|        std::cerr << "Something went wrong during " func ", please report this: " << e.what() \
  |  |  329|      0|                  << std::endl;                                                               \
  |  |  330|      0|        abort();                                                                              \
  |  |  331|      0|    }
  ------------------
  351|  7.42k|}
jsonnet_import_callback:
  379|  7.42k|{
  380|  7.42k|    vm->importCallback = cb;
  381|  7.42k|    vm->importCallbackContext = ctx;
  382|  7.42k|}
jsonnet_evaluate_snippet_stream:
  747|  7.42k|{
  748|  7.42k|    TRY
  ------------------
  |  |  319|  7.42k|#define TRY try {
  ------------------
  749|  7.42k|        return jsonnet_evaluate_snippet_aux(vm, filename, snippet, error, STREAM);
  750|  7.42k|    CATCH("jsonnet_evaluate_snippet_stream")
  ------------------
  |  |  321|  7.42k|    }                                                                                         \
  |  |  322|  7.42k|    catch (const std::bad_alloc &)                                                            \
  |  |  323|  7.42k|    {                                                                                         \
  |  |  324|      0|        memory_panic();                                                                       \
  |  |  325|      0|    }                                                                                         \
  |  |  326|  7.42k|    catch (const std::exception &e)                                                           \
  |  |  327|  7.42k|    {                                                                                         \
  |  |  328|      0|        std::cerr << "Something went wrong during " func ", please report this: " << e.what() \
  |  |  329|      0|                  << std::endl;                                                               \
  |  |  330|      0|        abort();                                                                              \
  |  |  331|      0|    }
  ------------------
  751|      0|    return nullptr;  // Never happens.
  752|  7.42k|}
jsonnet_realloc:
  755|  14.3k|{
  756|  14.3k|    (void)vm;
  757|  14.3k|    if (str == nullptr) {
  ------------------
  |  Branch (757:9): [True: 6.91k, False: 7.42k]
  ------------------
  758|  6.91k|        if (sz == 0)
  ------------------
  |  Branch (758:13): [True: 0, False: 6.91k]
  ------------------
  759|      0|            return nullptr;
  760|  6.91k|        auto *r = static_cast<char *>(::malloc(sz));
  761|  6.91k|        if (r == nullptr)
  ------------------
  |  Branch (761:13): [True: 0, False: 6.91k]
  ------------------
  762|      0|            memory_panic();
  763|  6.91k|        return r;
  764|  7.42k|    } else {
  765|  7.42k|        if (sz == 0) {
  ------------------
  |  Branch (765:13): [True: 7.42k, False: 0]
  ------------------
  766|  7.42k|            ::free(str);
  767|  7.42k|            return nullptr;
  768|  7.42k|        } else {
  769|      0|            auto *r = static_cast<char *>(::realloc(str, sz));
  770|      0|            if (r == nullptr)
  ------------------
  |  Branch (770:17): [True: 0, False: 0]
  ------------------
  771|      0|                memory_panic();
  772|      0|            return r;
  773|      0|        }
  774|  7.42k|    }
  775|  14.3k|}
_ZN9JsonnetVmC2Ev:
  228|  7.42k|        : gcGrowthTrigger(2.0),
  229|  7.42k|          maxStack(500),
  230|  7.42k|          gcMinObjects(1000),
  231|  7.42k|          maxTrace(20),
  232|  7.42k|          importCallback(default_import_callback),
  233|  7.42k|          importCallbackContext(this),
  234|  7.42k|          stringOutput(false),
  235|  7.42k|          trailingNewline(true),
  236|  7.42k|          fmtDebugDesugaring(false)
  237|  7.42k|    {
  238|  7.42k|        jpaths.emplace_back("/usr/share/jsonnet-" + std::string(jsonnet_version()) + "/");
  239|  7.42k|        jpaths.emplace_back("/usr/local/share/jsonnet-" + std::string(jsonnet_version()) + "/");
  240|  7.42k|    }
libjsonnet.cpp:_ZL11from_stringP9JsonnetVmRKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
   90|  6.91k|{
   91|  6.91k|    char *r = jsonnet_realloc(vm, nullptr, v.length() + 1);
   92|  6.91k|    std::memcpy(r, v.c_str(), v.length() + 1);
   93|  6.91k|    return r;
   94|  6.91k|}
libjsonnet.cpp:_ZL28jsonnet_evaluate_snippet_auxP9JsonnetVmPKcS2_PiN12_GLOBAL__N_18EvalKindE:
  541|  7.42k|{
  542|  7.42k|    try {
  543|  7.42k|        Allocator alloc;
  544|  7.42k|        AST *expr;
  545|  7.42k|        Tokens tokens = jsonnet_lex(filename, snippet);
  546|       |
  547|  7.42k|        expr = jsonnet_parse(&alloc, tokens);
  548|       |
  549|  7.42k|        jsonnet_desugar(&alloc, expr, &vm->tla);
  550|       |
  551|  7.42k|        unsigned max_stack = vm->maxStack;
  552|       |
  553|       |        // For the stdlib desugaring.
  554|  7.42k|        max_stack++;
  555|       |
  556|       |        // For the TLA desugaring.
  557|  7.42k|        max_stack++;
  558|       |
  559|  7.42k|        jsonnet_static_analysis(expr);
  560|  7.42k|        switch (kind) {
  561|      0|            case REGULAR: {
  ------------------
  |  Branch (561:13): [True: 0, False: 7.42k]
  ------------------
  562|      0|                std::string json_str = jsonnet_vm_execute(&alloc,
  563|      0|                                                          expr,
  564|      0|                                                          vm->ext,
  565|      0|                                                          max_stack,
  566|      0|                                                          vm->gcMinObjects,
  567|      0|                                                          vm->gcGrowthTrigger,
  568|      0|                                                          vm->nativeCallbacks,
  569|      0|                                                          vm->importCallback,
  570|      0|                                                          vm->importCallbackContext,
  571|      0|                                                          vm->stringOutput);
  572|      0|                if (vm->trailingNewline) {
  ------------------
  |  Branch (572:21): [True: 0, False: 0]
  ------------------
  573|      0|                    json_str += "\n";
  574|      0|                }
  575|      0|                *error = false;
  576|      0|                return from_string(vm, json_str);
  577|      0|            } break;
  578|       |
  579|      0|            case MULTI: {
  ------------------
  |  Branch (579:13): [True: 0, False: 7.42k]
  ------------------
  580|      0|                std::map<std::string, std::string> files =
  581|      0|                    jsonnet_vm_execute_multi(&alloc,
  582|      0|                                             expr,
  583|      0|                                             vm->ext,
  584|      0|                                             max_stack,
  585|      0|                                             vm->gcMinObjects,
  586|      0|                                             vm->gcGrowthTrigger,
  587|      0|                                             vm->nativeCallbacks,
  588|      0|                                             vm->importCallback,
  589|      0|                                             vm->importCallbackContext,
  590|      0|                                             vm->stringOutput);
  591|      0|                size_t sz = 1;  // final sentinel
  592|      0|                for (const auto &pair : files) {
  ------------------
  |  Branch (592:39): [True: 0, False: 0]
  ------------------
  593|      0|                    sz += pair.first.length() + 1;   // include sentinel
  594|      0|                    sz += pair.second.length() + 2;  // Add a '\n' as well as sentinel
  595|      0|                }
  596|      0|                char *buf = (char *)::malloc(sz);
  597|      0|                if (buf == nullptr)
  ------------------
  |  Branch (597:21): [True: 0, False: 0]
  ------------------
  598|      0|                    memory_panic();
  599|      0|                std::ptrdiff_t i = 0;
  600|      0|                for (const auto &pair : files) {
  ------------------
  |  Branch (600:39): [True: 0, False: 0]
  ------------------
  601|      0|                    memcpy(&buf[i], pair.first.c_str(), pair.first.length() + 1);
  602|      0|                    i += pair.first.length() + 1;
  603|      0|                    memcpy(&buf[i], pair.second.c_str(), pair.second.length());
  604|      0|                    i += pair.second.length();
  605|      0|                    if (vm->trailingNewline) {
  ------------------
  |  Branch (605:25): [True: 0, False: 0]
  ------------------
  606|      0|                        buf[i] = '\n';
  607|      0|                        i++;
  608|      0|                    }
  609|      0|                    buf[i] = '\0';
  610|      0|                    i++;
  611|      0|                }
  612|      0|                buf[i] = '\0';  // final sentinel
  613|      0|                *error = false;
  614|      0|                return buf;
  615|      0|            } break;
  616|       |
  617|  3.46k|            case STREAM: {
  ------------------
  |  Branch (617:13): [True: 3.46k, False: 3.96k]
  ------------------
  618|  3.46k|                if (!vm->trailingNewline) {
  ------------------
  |  Branch (618:21): [True: 0, False: 3.46k]
  ------------------
  619|      0|                    *error = true;
  620|      0|                    return from_string(vm, "INTERNAL: trailing-newline is required for streamed output");
  621|      0|                }
  622|  3.46k|                std::vector<std::string> documents =
  623|  3.46k|                    jsonnet_vm_execute_stream(&alloc,
  624|  3.46k|                                              expr,
  625|  3.46k|                                              vm->ext,
  626|  3.46k|                                              max_stack,
  627|  3.46k|                                              vm->gcMinObjects,
  628|  3.46k|                                              vm->gcGrowthTrigger,
  629|  3.46k|                                              vm->nativeCallbacks,
  630|  3.46k|                                              vm->importCallback,
  631|  3.46k|                                              vm->importCallbackContext,
  632|  3.46k|                                              vm->stringOutput);
  633|  3.46k|                size_t sz = 1;  // final sentinel
  634|   472k|                for (const auto &doc : documents) {
  ------------------
  |  Branch (634:38): [True: 472k, False: 3.46k]
  ------------------
  635|   472k|                    sz += doc.length() + 2;  // Add a '\n' as well as sentinel
  636|   472k|                }
  637|  3.46k|                char *buf = (char *)::malloc(sz);
  638|  3.46k|                if (buf == nullptr)
  ------------------
  |  Branch (638:21): [True: 0, False: 3.46k]
  ------------------
  639|      0|                    memory_panic();
  640|  3.46k|                std::ptrdiff_t i = 0;
  641|   472k|                for (const auto &doc : documents) {
  ------------------
  |  Branch (641:38): [True: 472k, False: 3.46k]
  ------------------
  642|   472k|                    memcpy(&buf[i], doc.c_str(), doc.length());
  643|   472k|                    i += doc.length();
  644|   472k|                    buf[i] = '\n';
  645|   472k|                    i++;
  646|   472k|                    buf[i] = '\0';
  647|   472k|                    i++;
  648|   472k|                }
  649|  3.46k|                buf[i] = '\0';  // final sentinel
  650|  3.46k|                *error = false;
  651|  3.46k|                return buf;
  652|  3.46k|            } break;
  653|       |
  654|      0|            default:
  ------------------
  |  Branch (654:13): [True: 0, False: 7.42k]
  ------------------
  655|      0|                fputs("INTERNAL ERROR: bad value of 'kind', probably memory corruption.\n", stderr);
  656|      0|                abort();
  657|  7.42k|        }
  658|       |
  659|  7.42k|    } catch (StaticError &e) {
  660|  3.96k|        std::stringstream ss;
  661|  3.96k|        ss << "STATIC ERROR: " << e << std::endl;
  662|  3.96k|        *error = true;
  663|  3.96k|        return from_string(vm, ss.str());
  664|       |
  665|  3.96k|    } catch (RuntimeError &e) {
  666|  2.95k|        std::stringstream ss;
  667|  2.95k|        ss << "RUNTIME ERROR: " << e.msg << std::endl;
  668|  2.95k|        const long max_above = vm->maxTrace / 2;
  669|  2.95k|        const long max_below = vm->maxTrace - max_above;
  670|  2.95k|        const long sz = e.stackTrace.size();
  671|   238k|        for (long i = 0; i < sz; ++i) {
  ------------------
  |  Branch (671:26): [True: 235k, False: 2.95k]
  ------------------
  672|   235k|            const auto &f = e.stackTrace[i];
  673|   235k|            if (vm->maxTrace > 0 && i >= max_above && i < sz - max_below) {
  ------------------
  |  Branch (673:17): [True: 235k, False: 0]
  |  Branch (673:37): [True: 223k, False: 11.7k]
  |  Branch (673:55): [True: 216k, False: 7.08k]
  ------------------
  674|   216k|                if (i == max_above)
  ------------------
  |  Branch (674:21): [True: 633, False: 215k]
  ------------------
  675|    633|                    ss << "\t..." << std::endl;
  676|   216k|            } else {
  677|  18.8k|                ss << "\t" << f.location << "\t" << f.name << std::endl;
  678|  18.8k|            }
  679|   235k|        }
  680|  2.95k|        *error = true;
  681|  2.95k|        return from_string(vm, ss.str());
  682|  2.95k|    }
  683|       |
  684|      0|    return nullptr;  // Quiet, compiler.
  685|  7.42k|}

_ZN7jsonnet8internal22jsonnet_unparse_numberEd:
   38|  1.73M|{
   39|  1.73M|    std::stringstream ss;
   40|       |    // Make sure we output the same thing, even if the user
   41|       |    // of the library changed the global locale
   42|  1.73M|    ss.imbue(std::locale::classic());
   43|  1.73M|    if (v == floor(v)) {
  ------------------
  |  Branch (43:9): [True: 1.73M, False: 8.85k]
  ------------------
   44|  1.73M|        ss << std::fixed << std::setprecision(0) << v;
   45|  1.73M|    } else {
   46|       |        // See "What Every Computer Scientist Should Know About Floating-Point Arithmetic"
   47|       |        // Theorem 15
   48|       |        // https://docs.oracle.com/cd/E19957-01/806-3568/ncg_goldberg.html
   49|  8.85k|        ss << std::setprecision(17);
   50|  8.85k|        ss << v;
   51|  8.85k|    }
   52|  1.73M|    return ss.str();
   53|  1.73M|}
_ZN7jsonnet8internal13jsonnet_parseEPNS0_9AllocatorERNSt3__14listINS0_5TokenENS3_9allocatorIS5_EEEE:
 1210|  14.4k|{
 1211|  14.4k|    Parser parser(tokens, alloc);
 1212|  14.4k|    unsigned parse_depth = 0;
 1213|  14.4k|    AST *expr = parser.parse(MAX_PRECEDENCE, parse_depth);
 1214|  14.4k|    if (tokens.front().kind != Token::END_OF_FILE) {
  ------------------
  |  Branch (1214:9): [True: 185, False: 14.2k]
  ------------------
 1215|    185|        std::stringstream ss;
 1216|    185|        ss << "did not expect: " << tokens.front();
 1217|    185|        throw StaticError(tokens.front().location, ss.str());
 1218|    185|    }
 1219|       |
 1220|  14.2k|    return expr;
 1221|  14.4k|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16ParserC2ERNSt3__14listINS0_5TokenENS3_9allocatorIS5_EEEEPNS0_9AllocatorE:
  155|  14.4k|    Parser(Tokens &tokens, Allocator *alloc) : tokens(tokens), alloc(alloc) {}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser5parseEjj:
  980|  39.5M|    {
  981|  39.5M|        if (current_depth >= MAX_PARSER_DEPTH) {
  ------------------
  |  Branch (981:13): [True: 3, False: 39.5M]
  ------------------
  982|      3|            throw StaticError(peek().location, "Exceeded maximum parse depth limit.");
  983|      3|        }
  984|       |
  985|  39.5M|        AST *ast = maybeParseGreedy(current_depth + 1);
  986|       |        // There cannot be an operator after a greedy parse.
  987|  39.5M|        if (ast != nullptr) return ast;
  ------------------
  |  Branch (987:13): [True: 5.25M, False: 34.2M]
  ------------------
  988|       |
  989|       |        // If we get here, we could be parsing an infix construct.
  990|       |
  991|       |        // Allocate this on the heap to control stack growth.
  992|  34.2M|        std::unique_ptr<Token> begin_(new Token(peek()));
  993|  34.2M|        const Token &begin = *begin_;
  994|       |
  995|  34.2M|        AST *lhs = parseTerminalBracketsOrUnary(current_depth + 1);
  996|       |
  997|  34.2M|        return parseInfix(lhs, begin, max_precedence, current_depth + 1);
  998|  39.5M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser4peekEv:
  123|   327M|    {
  124|   327M|        return tokens.front();
  125|   327M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser16maybeParseGreedyEj:
  811|  39.5M|    {
  812|  39.5M|        if (current_depth >= MAX_PARSER_DEPTH) {
  ------------------
  |  Branch (812:13): [True: 5, False: 39.5M]
  ------------------
  813|      5|            throw StaticError(peek().location, "Exceeded maximum parse depth limit.");
  814|      5|        }
  815|       |
  816|       |        // Allocate this on the heap to control stack growth.
  817|  39.5M|        std::unique_ptr<Token> begin_(new Token(peek()));
  818|  39.5M|        const Token &begin = *begin_;
  819|       |
  820|  39.5M|        switch (begin.kind) {
  821|       |            // These cases have effectively MAX_PRECEDENCE as the first
  822|       |            // call to parse will parse them.
  823|   311k|            case Token::ASSERT: {
  ------------------
  |  Branch (823:13): [True: 311k, False: 39.2M]
  ------------------
  824|   311k|                pop();
  825|   311k|                AST *cond = parse(MAX_PRECEDENCE, current_depth + 1);
  826|   311k|                Fodder colonFodder;
  827|   311k|                AST *msg = nullptr;
  828|   311k|                if (peek().kind == Token::OPERATOR && peek().data == ":") {
  ------------------
  |  Branch (828:21): [True: 287k, False: 23.5k]
  |  Branch (828:55): [True: 287k, False: 1]
  ------------------
  829|   287k|                    Token colon = pop();
  830|   287k|                    colonFodder = colon.fodder;
  831|   287k|                    msg = parse(MAX_PRECEDENCE, current_depth + 1);
  832|   287k|                }
  833|   311k|                Token semicolon = popExpect(Token::SEMICOLON);
  834|   311k|                AST *rest = parse(MAX_PRECEDENCE, current_depth + 1);
  835|   311k|                return alloc->make<Assert>(span(begin, rest),
  836|   311k|                                           begin.fodder,
  837|   311k|                                           cond,
  838|   311k|                                           colonFodder,
  839|   311k|                                           msg,
  840|   311k|                                           semicolon.fodder,
  841|   311k|                                           rest);
  842|      0|            }
  843|       |
  844|   493k|            case Token::ERROR: {
  ------------------
  |  Branch (844:13): [True: 493k, False: 39.0M]
  ------------------
  845|   493k|                pop();
  846|   493k|                AST *expr = parse(MAX_PRECEDENCE, current_depth + 1);
  847|   493k|                return alloc->make<Error>(span(begin, expr), begin.fodder, expr);
  848|      0|            }
  849|       |
  850|  2.46M|            case Token::IF: {
  ------------------
  |  Branch (850:13): [True: 2.46M, False: 37.0M]
  ------------------
  851|  2.46M|                pop();
  852|  2.46M|                AST *cond = parse(MAX_PRECEDENCE, current_depth + 1);
  853|  2.46M|                Token then = popExpect(Token::THEN);
  854|  2.46M|                AST *branch_true = parse(MAX_PRECEDENCE, current_depth + 1);
  855|  2.46M|                if (peek().kind == Token::ELSE) {
  ------------------
  |  Branch (855:21): [True: 2.43M, False: 30.7k]
  ------------------
  856|  2.43M|                    Token else_ = pop();
  857|  2.43M|                    AST *branch_false = parse(MAX_PRECEDENCE, current_depth + 1);
  858|  2.43M|                    return alloc->make<Conditional>(span(begin, branch_false),
  859|  2.43M|                                                    begin.fodder,
  860|  2.43M|                                                    cond,
  861|  2.43M|                                                    then.fodder,
  862|  2.43M|                                                    branch_true,
  863|  2.43M|                                                    else_.fodder,
  864|  2.43M|                                                    branch_false);
  865|  2.43M|                }
  866|  30.7k|                return alloc->make<Conditional>(span(begin, branch_true),
  867|  30.7k|                                                begin.fodder,
  868|  30.7k|                                                cond,
  869|  30.7k|                                                then.fodder,
  870|  30.7k|                                                branch_true,
  871|  30.7k|                                                Fodder{},
  872|  30.7k|                                                nullptr);
  873|  2.46M|            }
  874|       |
  875|   152k|            case Token::FUNCTION: {
  ------------------
  |  Branch (875:13): [True: 152k, False: 39.3M]
  ------------------
  876|   152k|                pop();  // Still available in 'begin'.
  877|   152k|                Token paren_l = pop();
  878|   152k|                if (paren_l.kind == Token::PAREN_L) {
  ------------------
  |  Branch (878:21): [True: 152k, False: 4]
  ------------------
  879|   152k|                    std::vector<AST *> params_asts;
  880|   152k|                    bool got_comma;
  881|   152k|                    Fodder paren_r_fodder;
  882|   152k|                    ArgParams params = parseParams("function parameter", got_comma, paren_r_fodder, current_depth);
  883|   152k|                    AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  884|   152k|                    return alloc->make<Function>(span(begin, body),
  885|   152k|                                                 begin.fodder,
  886|   152k|                                                 paren_l.fodder,
  887|   152k|                                                 params,
  888|   152k|                                                 got_comma,
  889|   152k|                                                 paren_r_fodder,
  890|   152k|                                                 body);
  891|   152k|                } else {
  892|      4|                    std::stringstream ss;
  893|      4|                    ss << "expected ( but got " << paren_l;
  894|      4|                    throw StaticError(paren_l.location, ss.str());
  895|      4|                }
  896|   152k|            }
  897|       |
  898|  1.08k|            case Token::IMPORT: {
  ------------------
  |  Branch (898:13): [True: 1.08k, False: 39.5M]
  ------------------
  899|  1.08k|                pop();
  900|  1.08k|                AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  901|  1.08k|                if (body->type == AST_LITERAL_STRING) {
  ------------------
  |  Branch (901:21): [True: 730, False: 356]
  ------------------
  902|    730|                    auto *lit = static_cast<LiteralString *>(body);
  903|    730|                    if (lit->tokenKind == LiteralString::BLOCK) {
  ------------------
  |  Branch (903:25): [True: 1, False: 729]
  ------------------
  904|      1|                        throw StaticError(lit->location,
  905|      1|                                          "Cannot use text blocks in import statements.");
  906|      1|                    }
  907|    729|                    return alloc->make<Import>(span(begin, body), begin.fodder, lit);
  908|    730|                } else {
  909|    356|                    std::stringstream ss;
  910|    356|                    ss << "computed imports are not allowed.";
  911|    356|                    throw StaticError(body->location, ss.str());
  912|    356|                }
  913|  1.08k|            }
  914|       |
  915|    874|            case Token::IMPORTSTR: {
  ------------------
  |  Branch (915:13): [True: 874, False: 39.5M]
  ------------------
  916|    874|                pop();
  917|    874|                AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  918|    874|                if (body->type == AST_LITERAL_STRING) {
  ------------------
  |  Branch (918:21): [True: 571, False: 303]
  ------------------
  919|    571|                    auto *lit = static_cast<LiteralString *>(body);
  920|    571|                    if (lit->tokenKind == LiteralString::BLOCK) {
  ------------------
  |  Branch (920:25): [True: 1, False: 570]
  ------------------
  921|      1|                        throw StaticError(lit->location,
  922|      1|                                          "Cannot use text blocks in import statements.");
  923|      1|                    }
  924|    570|                    return alloc->make<Importstr>(span(begin, body), begin.fodder, lit);
  925|    571|                } else {
  926|    303|                    std::stringstream ss;
  927|    303|                    ss << "computed imports are not allowed.";
  928|    303|                    throw StaticError(body->location, ss.str());
  929|    303|                }
  930|    874|            }
  931|       |
  932|    641|            case Token::IMPORTBIN: {
  ------------------
  |  Branch (932:13): [True: 641, False: 39.5M]
  ------------------
  933|    641|                pop();
  934|    641|                AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  935|    641|                if (body->type == AST_LITERAL_STRING) {
  ------------------
  |  Branch (935:21): [True: 417, False: 224]
  ------------------
  936|    417|                    auto *lit = static_cast<LiteralString *>(body);
  937|    417|                    if (lit->tokenKind == LiteralString::BLOCK) {
  ------------------
  |  Branch (937:25): [True: 0, False: 417]
  ------------------
  938|      0|                        throw StaticError(lit->location,
  939|      0|                                          "Cannot use text blocks in import statements.");
  940|      0|                    }
  941|    417|                    return alloc->make<Importbin>(span(begin, body), begin.fodder, lit);
  942|    417|                } else {
  943|    224|                    std::stringstream ss;
  944|    224|                    ss << "computed imports are not allowed.";
  945|    224|                    throw StaticError(body->location, ss.str());
  946|    224|                }
  947|    641|            }
  948|       |
  949|  1.83M|            case Token::LOCAL: {
  ------------------
  |  Branch (949:13): [True: 1.83M, False: 37.7M]
  ------------------
  950|  1.83M|                pop();
  951|  1.83M|                Local::Binds binds;
  952|  1.93M|                do {
  953|  1.93M|                    Token delim = parseBind(binds, current_depth + 1);
  954|  1.93M|                    if (delim.kind != Token::SEMICOLON && delim.kind != Token::COMMA) {
  ------------------
  |  Branch (954:25): [True: 100k, False: 1.83M]
  |  Branch (954:59): [True: 14, False: 100k]
  ------------------
  955|     14|                        std::stringstream ss;
  956|     14|                        ss << "expected , or ; but got " << delim;
  957|     14|                        throw StaticError(delim.location, ss.str());
  958|     14|                    }
  959|  1.93M|                    if (delim.kind == Token::SEMICOLON)
  ------------------
  |  Branch (959:25): [True: 1.82M, False: 100k]
  ------------------
  960|  1.82M|                        break;
  961|  1.93M|                } while (true);
  ------------------
  |  Branch (961:26): [True: 100k, Folded]
  ------------------
  962|  1.83M|                AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  963|  1.83M|                return alloc->make<Local>(span(begin, body), begin.fodder, binds, body);
  964|  1.83M|            }
  965|       |
  966|  34.2M|            default:
  ------------------
  |  Branch (966:13): [True: 34.2M, False: 5.25M]
  ------------------
  967|  34.2M|            return nullptr;
  968|  39.5M|        }
  969|  39.5M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser3popEv:
  111|   102M|    {
  112|   102M|        Token tok = peek();
  113|   102M|        tokens.pop_front();
  114|   102M|        return tok;
  115|   102M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser9popExpectENS0_5Token4KindEPKc:
  136|  16.3M|    {
  137|  16.3M|        Token tok = pop();
  138|  16.3M|        if (tok.kind != k) {
  ------------------
  |  Branch (138:13): [True: 242, False: 16.3M]
  ------------------
  139|    242|            std::stringstream ss;
  140|    242|            ss << "expected token " << k << " but got " << tok;
  141|    242|            throw StaticError(tok.location, ss.str());
  142|    242|        }
  143|  16.3M|        if (data != nullptr && tok.data != data) {
  ------------------
  |  Branch (143:13): [True: 1.98M, False: 14.3M]
  |  Branch (143:32): [True: 12, False: 1.98M]
  ------------------
  144|     12|            std::stringstream ss;
  145|     12|            ss << "expected operator " << data << " but got " << tok.data;
  146|     12|            throw StaticError(tok.location, ss.str());
  147|     12|        }
  148|  16.3M|        return tok;
  149|  16.3M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14spanERKNS0_5TokenEPNS0_3ASTE:
   94|  12.3M|{
   95|  12.3M|    return LocationRange(begin.location.file, begin.location.begin, end->location.end);
   96|  12.3M|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser11parseParamsERKNSt3__112basic_stringIcNS3_11char_traitsIcEENS3_9allocatorIcEEEERbRNS3_6vectorINS0_13FodderElementENS7_ISE_EEEEj:
  218|  1.66M|    {
  219|  1.66M|        ArgParams params;
  220|  1.66M|        Token paren_r = parseArgs(params, element_kind, got_comma, current_depth);
  221|       |
  222|       |        // Check they're all identifiers
  223|       |        // parseArgs returns f(x) with x as an expression.  Convert it here.
  224|  3.24M|        for (auto &p : params) {
  ------------------
  |  Branch (224:22): [True: 3.24M, False: 1.66M]
  ------------------
  225|  3.24M|            if (p.id == nullptr) {
  ------------------
  |  Branch (225:17): [True: 3.08M, False: 159k]
  ------------------
  226|  3.08M|                if (p.expr->type != AST_VAR) {
  ------------------
  |  Branch (226:21): [True: 2, False: 3.08M]
  ------------------
  227|      2|                    throw StaticError(p.expr->location, "could not parse parameter here.");
  228|      2|                }
  229|  3.08M|                auto *pv = static_cast<Var *>(p.expr);
  230|  3.08M|                p.id = pv->id;
  231|  3.08M|                p.idFodder = pv->openFodder;
  232|  3.08M|                p.expr = nullptr;
  233|  3.08M|            }
  234|  3.24M|        }
  235|       |
  236|  1.66M|        close_fodder = paren_r.fodder;
  237|       |
  238|  1.66M|        return params;
  239|  1.66M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser9parseArgsERNSt3__16vectorINS0_8ArgParamENS3_9allocatorIS5_EEEERKNS3_12basic_stringIcNS3_11char_traitsIcEENS6_IcEEEERbj:
  167|  7.80M|    {
  168|  7.80M|        got_comma = false;
  169|  7.80M|        bool first = true;
  170|  21.2M|        do {
  171|  21.2M|            Token next = peek();
  172|  21.2M|            if (next.kind == Token::PAREN_R) {
  ------------------
  |  Branch (172:17): [True: 7.79M, False: 13.4M]
  ------------------
  173|       |                // got_comma can be true or false here.
  174|  7.79M|                return pop();
  175|  7.79M|            }
  176|  13.4M|            if (!first && !got_comma) {
  ------------------
  |  Branch (176:17): [True: 5.64M, False: 7.79M]
  |  Branch (176:27): [True: 75, False: 5.64M]
  ------------------
  177|     75|                std::stringstream ss;
  178|     75|                ss << "expected a comma before next " << element_kind << ".";
  179|     75|                throw StaticError(next.location, ss.str());
  180|     75|            }
  181|       |            // Either id=expr or id or expr, but note that expr could be id==1 so this needs
  182|       |            // look-ahead.
  183|  13.4M|            Fodder id_fodder;
  184|  13.4M|            const Identifier *id = nullptr;
  185|  13.4M|            Fodder eq_fodder;
  186|  13.4M|            if (peek().kind == Token::IDENTIFIER) {
  ------------------
  |  Branch (186:17): [True: 11.5M, False: 1.86M]
  ------------------
  187|  11.5M|                Token maybe_eq = doublePeek();
  188|  11.5M|                if (maybe_eq.kind == Token::OPERATOR && maybe_eq.data == "=") {
  ------------------
  |  Branch (188:21): [True: 1.20M, False: 10.3M]
  |  Branch (188:57): [True: 185k, False: 1.02M]
  ------------------
  189|   185k|                    id_fodder = peek().fodder;
  190|   185k|                    id = alloc->makeIdentifier(peek().data32());
  191|   185k|                    eq_fodder = maybe_eq.fodder;
  192|   185k|                    pop();  // id
  193|   185k|                    pop();  // eq
  194|   185k|                }
  195|  11.5M|            }
  196|  13.4M|            AST *expr = parse(MAX_PRECEDENCE, current_depth + 1);
  197|  13.4M|            got_comma = false;
  198|  13.4M|            first = false;
  199|  13.4M|            Fodder comma_fodder;
  200|  13.4M|            if (peek().kind == Token::COMMA) {
  ------------------
  |  Branch (200:17): [True: 5.64M, False: 7.79M]
  ------------------
  201|  5.64M|                Token comma = pop();
  202|  5.64M|                comma_fodder = comma.fodder;
  203|  5.64M|                got_comma = true;
  204|  5.64M|            }
  205|  13.4M|            args.emplace_back(id_fodder, id, eq_fodder, expr, comma_fodder);
  206|  13.4M|        } while (true);
  ------------------
  |  Branch (206:18): [True: 13.4M, Folded]
  ------------------
  207|  7.80M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser10doublePeekEv:
  129|  11.5M|    {
  130|  11.5M|        Tokens::iterator it = tokens.begin();  // First one.
  131|  11.5M|        it++;                                  // Now pointing at the second one.
  132|  11.5M|        return *(it);
  133|  11.5M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser9parseBindERNSt3__16vectorINS0_5Local4BindENS3_9allocatorIS6_EEEEj:
  248|  1.93M|    {
  249|  1.93M|        Token var_id = popExpect(Token::IDENTIFIER);
  250|  1.93M|        auto *id = alloc->makeIdentifier(var_id.data32());
  251|  1.93M|        for (const auto &bind : binds) {
  ------------------
  |  Branch (251:31): [True: 272k, False: 1.93M]
  ------------------
  252|   272k|            if (bind.var == id)
  ------------------
  |  Branch (252:17): [True: 5, False: 272k]
  ------------------
  253|      5|                throw StaticError(var_id.location, "duplicate local var: " + var_id.data);
  254|   272k|        }
  255|  1.93M|        bool is_function = false;
  256|  1.93M|        ArgParams params;
  257|  1.93M|        bool trailing_comma = false;
  258|  1.93M|        Fodder fodder_l, fodder_r;
  259|  1.93M|        if (peek().kind == Token::PAREN_L) {
  ------------------
  |  Branch (259:13): [True: 559k, False: 1.37M]
  ------------------
  260|   559k|            Token paren_l = pop();
  261|   559k|            fodder_l = paren_l.fodder;
  262|   559k|            params = parseParams("function parameter", trailing_comma, fodder_r, current_depth);
  263|   559k|            is_function = true;
  264|   559k|        }
  265|  1.93M|        Token eq = popExpect(Token::OPERATOR, "=");
  266|  1.93M|        AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  267|  1.93M|        Token delim = pop();
  268|  1.93M|        binds.emplace_back(var_id.fodder,
  269|  1.93M|                           id,
  270|  1.93M|                           eq.fodder,
  271|  1.93M|                           body,
  272|  1.93M|                           is_function,
  273|  1.93M|                           fodder_l,
  274|  1.93M|                           params,
  275|  1.93M|                           trailing_comma,
  276|  1.93M|                           fodder_r,
  277|  1.93M|                           delim.fodder);
  278|  1.93M|        return delim;
  279|  1.93M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser28parseTerminalBracketsOrUnaryEj:
  627|  34.2M|    {
  628|  34.2M|        if (current_depth >= MAX_PARSER_DEPTH) {
  ------------------
  |  Branch (628:13): [True: 0, False: 34.2M]
  ------------------
  629|      0|            throw StaticError(peek().location, "Exceeded maximum parse depth limit.");
  630|      0|        }
  631|       |
  632|  34.2M|        Token tok = pop();
  633|  34.2M|        switch (tok.kind) {
  ------------------
  |  Branch (633:17): [True: 34.2M, False: 0]
  ------------------
  634|      0|            case Token::ASSERT:
  ------------------
  |  Branch (634:13): [True: 0, False: 34.2M]
  ------------------
  635|     10|            case Token::BRACE_R:
  ------------------
  |  Branch (635:13): [True: 10, False: 34.2M]
  ------------------
  636|     23|            case Token::BRACKET_R:
  ------------------
  |  Branch (636:13): [True: 13, False: 34.2M]
  ------------------
  637|     50|            case Token::COMMA:
  ------------------
  |  Branch (637:13): [True: 27, False: 34.2M]
  ------------------
  638|     60|            case Token::DOT:
  ------------------
  |  Branch (638:13): [True: 10, False: 34.2M]
  ------------------
  639|     61|            case Token::ELSE:
  ------------------
  |  Branch (639:13): [True: 1, False: 34.2M]
  ------------------
  640|     61|            case Token::ERROR:
  ------------------
  |  Branch (640:13): [True: 0, False: 34.2M]
  ------------------
  641|     62|            case Token::FOR:
  ------------------
  |  Branch (641:13): [True: 1, False: 34.2M]
  ------------------
  642|     62|            case Token::FUNCTION:
  ------------------
  |  Branch (642:13): [True: 0, False: 34.2M]
  ------------------
  643|     62|            case Token::IF:
  ------------------
  |  Branch (643:13): [True: 0, False: 34.2M]
  ------------------
  644|     66|            case Token::IN:
  ------------------
  |  Branch (644:13): [True: 4, False: 34.2M]
  ------------------
  645|     66|            case Token::IMPORT:
  ------------------
  |  Branch (645:13): [True: 0, False: 34.2M]
  ------------------
  646|     66|            case Token::IMPORTSTR:
  ------------------
  |  Branch (646:13): [True: 0, False: 34.2M]
  ------------------
  647|     66|            case Token::IMPORTBIN:
  ------------------
  |  Branch (647:13): [True: 0, False: 34.2M]
  ------------------
  648|     66|            case Token::LOCAL:
  ------------------
  |  Branch (648:13): [True: 0, False: 34.2M]
  ------------------
  649|     76|            case Token::PAREN_R:
  ------------------
  |  Branch (649:13): [True: 10, False: 34.2M]
  ------------------
  650|     84|            case Token::SEMICOLON:
  ------------------
  |  Branch (650:13): [True: 8, False: 34.2M]
  ------------------
  651|     86|            case Token::TAILSTRICT:
  ------------------
  |  Branch (651:13): [True: 2, False: 34.2M]
  ------------------
  652|     87|            case Token::THEN: throw unexpected(tok, "parsing terminal");
  ------------------
  |  Branch (652:13): [True: 1, False: 34.2M]
  ------------------
  653|       |
  654|    693|            case Token::END_OF_FILE: throw StaticError(tok.location, "unexpected end of file.");
  ------------------
  |  Branch (654:13): [True: 693, False: 34.2M]
  ------------------
  655|       |
  656|   508k|            case Token::OPERATOR: {
  ------------------
  |  Branch (656:13): [True: 508k, False: 33.7M]
  ------------------
  657|   508k|                UnaryOp uop;
  658|   508k|                if (!op_is_unary(tok.data, uop)) {
  ------------------
  |  Branch (658:21): [True: 233, False: 508k]
  ------------------
  659|    233|                    std::stringstream ss;
  660|    233|                    ss << "not a unary operator: " << tok.data;
  661|    233|                    throw StaticError(tok.location, ss.str());
  662|    233|                }
  663|   508k|                AST *expr = parse(UNARY_PRECEDENCE, current_depth + 1);
  664|   508k|                return alloc->make<Unary>(span(tok, expr), tok.fodder, uop, expr);
  665|   508k|            }
  666|   361k|            case Token::BRACE_L: {
  ------------------
  |  Branch (666:13): [True: 361k, False: 33.9M]
  ------------------
  667|   361k|                AST *obj;
  668|   361k|                parseObjectRemainder(obj, tok, current_depth + 1);
  669|   361k|                return obj;
  670|   508k|            }
  671|       |
  672|  1.04M|            case Token::BRACKET_L: {
  ------------------
  |  Branch (672:13): [True: 1.04M, False: 33.2M]
  ------------------
  673|  1.04M|                Token next = peek();
  674|  1.04M|                if (next.kind == Token::BRACKET_R) {
  ------------------
  |  Branch (674:21): [True: 172k, False: 876k]
  ------------------
  675|   172k|                    Token bracket_r = pop();
  676|   172k|                    return alloc->make<Array>(
  677|   172k|                        span(tok, next), tok.fodder, Array::Elements{}, false, bracket_r.fodder);
  678|   172k|                }
  679|   876k|                AST *first = parse(MAX_PRECEDENCE, current_depth + 1);
  680|   876k|                bool got_comma = false;
  681|   876k|                Fodder comma_fodder;
  682|   876k|                next = peek();
  683|   876k|                if (!got_comma && next.kind == Token::COMMA) {
  ------------------
  |  Branch (683:21): [True: 874k, False: 2.02k]
  |  Branch (683:35): [True: 134k, False: 740k]
  ------------------
  684|   134k|                    Token comma = pop();
  685|   134k|                    comma_fodder = comma.fodder;
  686|   134k|                    next = peek();
  687|   134k|                    got_comma = true;
  688|   134k|                }
  689|       |
  690|   876k|                if (next.kind == Token::FOR) {
  ------------------
  |  Branch (690:21): [True: 242k, False: 634k]
  ------------------
  691|       |                    // It's a comprehension
  692|   242k|                    Token for_token = pop();
  693|   242k|                    std::vector<ComprehensionSpec> specs;
  694|   242k|                    Token last = parseComprehensionSpecs(Token::BRACKET_R, for_token.fodder, specs, current_depth + 1);
  695|   242k|                    return alloc->make<ArrayComprehension>(span(tok, last),
  696|   242k|                                                           tok.fodder,
  697|   242k|                                                           first,
  698|   242k|                                                           comma_fodder,
  699|   242k|                                                           got_comma,
  700|   242k|                                                           specs,
  701|   242k|                                                           last.fodder);
  702|   242k|                }
  703|       |
  704|       |                // Not a comprehension: It can have more elements.
  705|   634k|                Array::Elements elements;
  706|   634k|                elements.emplace_back(first, comma_fodder);
  707|  1.96M|                do {
  708|  1.96M|                    if (next.kind == Token::BRACKET_R) {
  ------------------
  |  Branch (708:25): [True: 630k, False: 1.33M]
  ------------------
  709|   630k|                        Token bracket_r = pop();
  710|   630k|                        return alloc->make<Array>(
  711|   630k|                            span(tok, next), tok.fodder, elements, got_comma, bracket_r.fodder);
  712|   630k|                    }
  713|  1.33M|                    if (!got_comma) {
  ------------------
  |  Branch (713:25): [True: 275, False: 1.33M]
  ------------------
  714|    275|                        std::stringstream ss;
  715|    275|                        ss << "expected a comma before next array element.";
  716|    275|                        throw StaticError(next.location, ss.str());
  717|    275|                    }
  718|  1.33M|                    AST *expr = parse(MAX_PRECEDENCE, current_depth + 1);
  719|  1.33M|                    comma_fodder.clear();
  720|  1.33M|                    got_comma = false;
  721|  1.33M|                    next = peek();
  722|  1.33M|                    if (next.kind == Token::COMMA) {
  ------------------
  |  Branch (722:25): [True: 1.21M, False: 120k]
  ------------------
  723|  1.21M|                        Token comma = pop();
  724|  1.21M|                        comma_fodder = comma.fodder;
  725|  1.21M|                        next = peek();
  726|  1.21M|                        got_comma = true;
  727|  1.21M|                    }
  728|  1.33M|                    elements.emplace_back(expr, comma_fodder);
  729|  1.33M|                } while (true);
  ------------------
  |  Branch (729:26): [True: 1.32M, Folded]
  ------------------
  730|   634k|            }
  731|       |
  732|   382k|            case Token::PAREN_L: {
  ------------------
  |  Branch (732:13): [True: 379k, False: 33.9M]
  ------------------
  733|   382k|                auto *inner = parse(MAX_PRECEDENCE, current_depth + 1);
  734|   382k|                Token close = popExpect(Token::PAREN_R);
  735|   382k|                return alloc->make<Parens>(span(tok, close), tok.fodder, inner, close.fodder);
  736|   634k|            }
  737|       |
  738|       |            // Literals
  739|  4.03M|            case Token::NUMBER: return alloc->make<LiteralNumber>(span(tok), tok.fodder, tok.data);
  ------------------
  |  Branch (739:13): [True: 4.03M, False: 30.2M]
  ------------------
  740|       |
  741|  3.52M|            case Token::STRING_SINGLE:
  ------------------
  |  Branch (741:13): [True: 3.52M, False: 30.7M]
  ------------------
  742|  3.52M|                return alloc->make<LiteralString>(
  743|  3.52M|                    span(tok), tok.fodder, tok.data32(), LiteralString::SINGLE, "", "");
  744|  86.2k|            case Token::STRING_DOUBLE:
  ------------------
  |  Branch (744:13): [True: 86.2k, False: 34.2M]
  ------------------
  745|  86.2k|                return alloc->make<LiteralString>(
  746|  86.2k|                    span(tok), tok.fodder, tok.data32(), LiteralString::DOUBLE, "", "");
  747|  3.62k|            case Token::STRING_BLOCK:
  ------------------
  |  Branch (747:13): [True: 3.62k, False: 34.2M]
  ------------------
  748|  3.62k|                return alloc->make<LiteralString>(span(tok),
  749|  3.62k|                                                  tok.fodder,
  750|  3.62k|                                                  tok.data32(),
  751|  3.62k|                                                  LiteralString::BLOCK,
  752|  3.62k|                                                  tok.stringBlockIndent,
  753|  3.62k|                                                  tok.stringBlockTermIndent);
  754|  1.19k|            case Token::VERBATIM_STRING_SINGLE:
  ------------------
  |  Branch (754:13): [True: 1.19k, False: 34.2M]
  ------------------
  755|  1.19k|                return alloc->make<LiteralString>(
  756|  1.19k|                    span(tok), tok.fodder, tok.data32(), LiteralString::VERBATIM_SINGLE, "", "");
  757|    777|            case Token::VERBATIM_STRING_DOUBLE:
  ------------------
  |  Branch (757:13): [True: 777, False: 34.2M]
  ------------------
  758|    777|                return alloc->make<LiteralString>(
  759|    777|                    span(tok), tok.fodder, tok.data32(), LiteralString::VERBATIM_DOUBLE, "", "");
  760|       |
  761|   371k|            case Token::FALSE: return alloc->make<LiteralBoolean>(span(tok), tok.fodder, false);
  ------------------
  |  Branch (761:13): [True: 371k, False: 33.9M]
  ------------------
  762|       |
  763|   272k|            case Token::TRUE: return alloc->make<LiteralBoolean>(span(tok), tok.fodder, true);
  ------------------
  |  Branch (763:13): [True: 272k, False: 34.0M]
  ------------------
  764|       |
  765|   138k|            case Token::NULL_LIT: return alloc->make<LiteralNull>(span(tok), tok.fodder);
  ------------------
  |  Branch (765:13): [True: 138k, False: 34.1M]
  ------------------
  766|       |
  767|       |            // Variables
  768|  26.0k|            case Token::DOLLAR: return alloc->make<Dollar>(span(tok), tok.fodder);
  ------------------
  |  Branch (768:13): [True: 26.0k, False: 34.2M]
  ------------------
  769|       |
  770|  23.4M|            case Token::IDENTIFIER:
  ------------------
  |  Branch (770:13): [True: 23.4M, False: 10.8M]
  ------------------
  771|  23.4M|                return alloc->make<Var>(span(tok), tok.fodder, alloc->makeIdentifier(tok.data32()));
  772|       |
  773|  53.5k|            case Token::SELF: return alloc->make<Self>(span(tok), tok.fodder);
  ------------------
  |  Branch (773:13): [True: 53.5k, False: 34.2M]
  ------------------
  774|       |
  775|  3.15k|            case Token::SUPER: {
  ------------------
  |  Branch (775:13): [True: 3.15k, False: 34.2M]
  ------------------
  776|  3.15k|                Token next = pop();
  777|  3.15k|                AST *index = nullptr;
  778|  3.15k|                const Identifier *id = nullptr;
  779|  3.15k|                Fodder id_fodder;
  780|  3.15k|                switch (next.kind) {
  781|  2.84k|                    case Token::DOT: {
  ------------------
  |  Branch (781:21): [True: 2.84k, False: 307]
  ------------------
  782|  2.84k|                        Token field_id = popExpect(Token::IDENTIFIER);
  783|  2.84k|                        id_fodder = field_id.fodder;
  784|  2.84k|                        id = alloc->makeIdentifier(field_id.data32());
  785|  2.84k|                    } break;
  786|    304|                    case Token::BRACKET_L: {
  ------------------
  |  Branch (786:21): [True: 304, False: 2.84k]
  ------------------
  787|    304|                        index = parse(MAX_PRECEDENCE, current_depth + 1);
  788|    304|                        Token bracket_r = popExpect(Token::BRACKET_R);
  789|    304|                        id_fodder = bracket_r.fodder;  // Not id_fodder, but use the same var.
  790|    304|                    } break;
  791|      3|                    default: throw StaticError(tok.location, "expected . or [ after super.");
  ------------------
  |  Branch (791:21): [True: 3, False: 3.14k]
  ------------------
  792|  3.15k|                }
  793|  2.94k|                return alloc->make<SuperIndex>(
  794|  2.94k|                    span(tok), tok.fodder, next.fodder, index, id_fodder, id);
  795|  3.15k|            }
  796|  34.2M|        }
  797|       |
  798|      0|        std::cerr << "INTERNAL ERROR: Unknown tok kind: " << tok.kind << std::endl;
  799|      0|        std::abort();
  800|      0|        return nullptr;  // Quiet, compiler.
  801|  34.2M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser10unexpectedERKNS0_5TokenERKNSt3__112basic_stringIcNS6_11char_traitsIcEENS6_9allocatorIcEEEE:
  104|    319|    {
  105|    319|        std::stringstream ss;
  106|    319|        ss << "unexpected: " << tok.kind << " while " << while_;
  107|    319|        return StaticError(tok.location, ss.str());
  108|    319|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111op_is_unaryERKNSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEERNS0_7UnaryOpE:
   66|   508k|{
   67|   508k|    auto it = unary_map.find(op);
   68|   508k|    if (it == unary_map.end())
  ------------------
  |  Branch (68:9): [True: 233, False: 508k]
  ------------------
   69|    233|        return false;
   70|   508k|    uop = it->second;
   71|   508k|    return true;
   72|   508k|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser20parseObjectRemainderERPNS0_3ASTERKNS0_5TokenEj:
  289|   420k|    {
  290|   420k|        if (current_depth >= MAX_PARSER_DEPTH) {
  ------------------
  |  Branch (290:13): [True: 1, False: 420k]
  ------------------
  291|      1|            throw StaticError(peek().location, "Exceeded maximum parse depth limit.");
  292|      1|        }
  293|       |
  294|   420k|        ObjectFields fields;
  295|   420k|        std::set<std::string> literal_fields;  // For duplicate fields detection.
  296|   420k|        std::set<const Identifier *> binds;    // For duplicate locals detection.
  297|       |
  298|   420k|        bool got_comma = false;
  299|   420k|        bool first = true;
  300|   420k|        Token next = pop();
  301|       |
  302|  2.32M|        do {
  303|  2.32M|            if (next.kind == Token::BRACE_R) {
  ------------------
  |  Branch (303:17): [True: 378k, False: 1.94M]
  ------------------
  304|   378k|                obj = alloc->make<Object>(
  305|   378k|                    span(tok, next), tok.fodder, fields, got_comma, next.fodder);
  306|   378k|                return next;
  307|       |
  308|  1.94M|            } else if (next.kind == Token::FOR) {
  ------------------
  |  Branch (308:24): [True: 35.9k, False: 1.91M]
  ------------------
  309|       |                // It's a comprehension
  310|  35.9k|                unsigned num_fields = 0;
  311|  35.9k|                unsigned num_asserts = 0;
  312|  35.9k|                const ObjectField *field_ptr = nullptr;
  313|  36.1k|                for (const auto &field : fields) {
  ------------------
  |  Branch (313:40): [True: 36.1k, False: 35.9k]
  ------------------
  314|  36.1k|                    if (field.kind == ObjectField::LOCAL)
  ------------------
  |  Branch (314:25): [True: 1, False: 36.1k]
  ------------------
  315|      1|                        continue;
  316|  36.1k|                    if (field.kind == ObjectField::ASSERT) {
  ------------------
  |  Branch (316:25): [True: 261, False: 35.9k]
  ------------------
  317|    261|                        num_asserts++;
  318|    261|                        continue;
  319|    261|                    }
  320|  35.9k|                    field_ptr = &field;
  321|  35.9k|                    num_fields++;
  322|  35.9k|                }
  323|  35.9k|                if (num_asserts > 0) {
  ------------------
  |  Branch (323:21): [True: 9, False: 35.9k]
  ------------------
  324|      9|                    auto msg = "object comprehension cannot have asserts.";
  325|      9|                    throw StaticError(next.location, msg);
  326|      9|                }
  327|  35.9k|                if (num_fields != 1) {
  ------------------
  |  Branch (327:21): [True: 5, False: 35.9k]
  ------------------
  328|      5|                    auto msg = "object comprehension can only have one field.";
  329|      5|                    throw StaticError(next.location, msg);
  330|      5|                }
  331|  35.9k|                const ObjectField &field = *field_ptr;
  332|       |
  333|  35.9k|                if (field.hide != ObjectField::INHERIT) {
  ------------------
  |  Branch (333:21): [True: 1, False: 35.9k]
  ------------------
  334|      1|                    auto msg = "object comprehensions cannot have hidden fields.";
  335|      1|                    throw StaticError(next.location, msg);
  336|      1|                }
  337|       |
  338|  35.9k|                if (field.kind != ObjectField::FIELD_EXPR) {
  ------------------
  |  Branch (338:21): [True: 1, False: 35.9k]
  ------------------
  339|      1|                    auto msg = "object comprehensions can only have [e] fields.";
  340|      1|                    throw StaticError(next.location, msg);
  341|      1|                }
  342|       |
  343|  35.9k|                std::vector<ComprehensionSpec> specs;
  344|  35.9k|                Token last = parseComprehensionSpecs(Token::BRACE_R, next.fodder, specs, current_depth + 1);
  345|  35.9k|                obj = alloc->make<ObjectComprehension>(
  346|  35.9k|                    span(tok, last), tok.fodder, fields, got_comma, specs, last.fodder);
  347|       |
  348|  35.9k|                return last;
  349|  35.9k|            }
  350|       |
  351|  1.91M|            if (!got_comma && !first)
  ------------------
  |  Branch (351:17): [True: 382k, False: 1.52M]
  |  Branch (351:31): [True: 84, False: 382k]
  ------------------
  352|     84|                throw StaticError(next.location, "expected a comma before next field.");
  353|       |
  354|  1.91M|            first = false;
  355|  1.91M|            got_comma = false;
  356|       |
  357|  1.91M|            switch (next.kind) {
  358|  40.9k|                case Token::BRACKET_L:
  ------------------
  |  Branch (358:17): [True: 40.9k, False: 1.87M]
  ------------------
  359|  1.74M|                case Token::IDENTIFIER:
  ------------------
  |  Branch (359:17): [True: 1.70M, False: 208k]
  ------------------
  360|  1.78M|                case Token::STRING_DOUBLE:
  ------------------
  |  Branch (360:17): [True: 37.6k, False: 1.87M]
  ------------------
  361|  1.81M|                case Token::STRING_SINGLE:
  ------------------
  |  Branch (361:17): [True: 30.4k, False: 1.88M]
  ------------------
  362|  1.81M|                case Token::STRING_BLOCK:
  ------------------
  |  Branch (362:17): [True: 1.17k, False: 1.91M]
  ------------------
  363|  1.81M|                case Token::VERBATIM_STRING_DOUBLE:
  ------------------
  |  Branch (363:17): [True: 686, False: 1.91M]
  ------------------
  364|  1.81M|                case Token::VERBATIM_STRING_SINGLE: {
  ------------------
  |  Branch (364:17): [True: 485, False: 1.91M]
  ------------------
  365|  1.81M|                    ObjectField::Kind kind;
  366|  1.81M|                    AST *expr1 = nullptr;
  367|  1.81M|                    const Identifier *id = nullptr;
  368|  1.81M|                    Fodder fodder1, fodder2;
  369|  1.81M|                    LocationRange idLocation;
  370|  1.81M|                    if (next.kind == Token::IDENTIFIER) {
  ------------------
  |  Branch (370:25): [True: 1.70M, False: 111k]
  ------------------
  371|  1.70M|                        fodder1 = next.fodder;
  372|  1.70M|                        kind = ObjectField::FIELD_ID;
  373|  1.70M|                        id = alloc->makeIdentifier(next.data32());
  374|  1.70M|                        idLocation = next.location;
  375|  1.70M|                    } else if (next.kind == Token::STRING_DOUBLE) {
  ------------------
  |  Branch (375:32): [True: 37.6k, False: 73.7k]
  ------------------
  376|  37.6k|                        kind = ObjectField::FIELD_STR;
  377|  37.6k|                        expr1 = alloc->make<LiteralString>(next.location,
  378|  37.6k|                                                           next.fodder,
  379|  37.6k|                                                           next.data32(),
  380|  37.6k|                                                           LiteralString::DOUBLE,
  381|  37.6k|                                                           "",
  382|  37.6k|                                                           "");
  383|  73.7k|                    } else if (next.kind == Token::STRING_SINGLE) {
  ------------------
  |  Branch (383:32): [True: 30.4k, False: 43.2k]
  ------------------
  384|  30.4k|                        kind = ObjectField::FIELD_STR;
  385|  30.4k|                        expr1 = alloc->make<LiteralString>(next.location,
  386|  30.4k|                                                           next.fodder,
  387|  30.4k|                                                           next.data32(),
  388|  30.4k|                                                           LiteralString::SINGLE,
  389|  30.4k|                                                           "",
  390|  30.4k|                                                           "");
  391|  43.2k|                    } else if (next.kind == Token::STRING_BLOCK) {
  ------------------
  |  Branch (391:32): [True: 1.17k, False: 42.1k]
  ------------------
  392|  1.17k|                        kind = ObjectField::FIELD_STR;
  393|  1.17k|                        expr1 = alloc->make<LiteralString>(next.location,
  394|  1.17k|                                                           next.fodder,
  395|  1.17k|                                                           next.data32(),
  396|  1.17k|                                                           LiteralString::BLOCK,
  397|  1.17k|                                                           next.stringBlockIndent,
  398|  1.17k|                                                           next.stringBlockTermIndent);
  399|  42.1k|                    } else if (next.kind == Token::VERBATIM_STRING_SINGLE) {
  ------------------
  |  Branch (399:32): [True: 485, False: 41.6k]
  ------------------
  400|    485|                        kind = ObjectField::FIELD_STR;
  401|    485|                        expr1 = alloc->make<LiteralString>(next.location,
  402|    485|                                                           next.fodder,
  403|    485|                                                           next.data32(),
  404|    485|                                                           LiteralString::VERBATIM_SINGLE,
  405|    485|                                                           "",
  406|    485|                                                           "");
  407|  41.6k|                    } else if (next.kind == Token::VERBATIM_STRING_DOUBLE) {
  ------------------
  |  Branch (407:32): [True: 686, False: 40.9k]
  ------------------
  408|    686|                        kind = ObjectField::FIELD_STR;
  409|    686|                        expr1 = alloc->make<LiteralString>(next.location,
  410|    686|                                                           next.fodder,
  411|    686|                                                           next.data32(),
  412|    686|                                                           LiteralString::VERBATIM_DOUBLE,
  413|    686|                                                           "",
  414|    686|                                                           "");
  415|  40.9k|                    } else {
  416|  40.9k|                        kind = ObjectField::FIELD_EXPR;
  417|  40.9k|                        fodder1 = next.fodder;
  418|  40.9k|                        expr1 = parse(MAX_PRECEDENCE, current_depth + 1);
  419|  40.9k|                        Token bracket_r = popExpect(Token::BRACKET_R);
  420|  40.9k|                        fodder2 = bracket_r.fodder;
  421|  40.9k|                    }
  422|       |
  423|  1.81M|                    bool is_method = false;
  424|  1.81M|                    bool meth_comma = false;
  425|  1.81M|                    ArgParams params;
  426|  1.81M|                    Fodder fodder_l;
  427|  1.81M|                    Fodder fodder_r;
  428|  1.81M|                    if (peek().kind == Token::PAREN_L) {
  ------------------
  |  Branch (428:25): [True: 940k, False: 873k]
  ------------------
  429|   940k|                        Token paren_l = pop();
  430|   940k|                        fodder_l = paren_l.fodder;
  431|   940k|                        params = parseParams("method parameter", meth_comma, fodder_r, current_depth);
  432|   940k|                        is_method = true;
  433|   940k|                    }
  434|       |
  435|  1.81M|                    bool plus_sugar = false;
  436|       |
  437|  1.81M|                    Token op = popExpect(Token::OPERATOR);
  438|  1.81M|                    const char *od = op.data.c_str();
  439|  1.81M|                    if (*od == '+') {
  ------------------
  |  Branch (439:25): [True: 14.8k, False: 1.79M]
  ------------------
  440|  14.8k|                        plus_sugar = true;
  441|  14.8k|                        od++;
  442|  14.8k|                    }
  443|  1.81M|                    unsigned colons = 0;
  444|  4.57M|                    for (; *od != '\0'; ++od) {
  ------------------
  |  Branch (444:28): [True: 2.75M, False: 1.81M]
  ------------------
  445|  2.75M|                        if (*od != ':') {
  ------------------
  |  Branch (445:29): [True: 16, False: 2.75M]
  ------------------
  446|     16|                            throw StaticError(
  447|     16|                                next.location,
  448|     16|                                "expected one of :, ::, :::, +:, +::, +:::, got: " + op.data);
  449|     16|                        }
  450|  2.75M|                        ++colons;
  451|  2.75M|                    }
  452|  1.81M|                    ObjectField::Hide field_hide;
  453|  1.81M|                    switch (colons) {
  454|   867k|                        case 1: field_hide = ObjectField::INHERIT; break;
  ------------------
  |  Branch (454:25): [True: 867k, False: 947k]
  ------------------
  455|       |
  456|   945k|                        case 2: field_hide = ObjectField::HIDDEN; break;
  ------------------
  |  Branch (456:25): [True: 945k, False: 869k]
  ------------------
  457|       |
  458|      4|                        case 3: field_hide = ObjectField::VISIBLE; break;
  ------------------
  |  Branch (458:25): [True: 4, False: 1.81M]
  ------------------
  459|       |
  460|     12|                        default:
  ------------------
  |  Branch (460:25): [True: 12, False: 1.81M]
  ------------------
  461|     12|                            throw StaticError(
  462|     12|                                next.location,
  463|     12|                                "expected one of :, ::, :::, +:, +::, +:::, got: " + op.data);
  464|  1.81M|                    }
  465|       |
  466|       |                    // Basic checks for invalid Jsonnet code.
  467|  1.81M|                    if (is_method && plus_sugar) {
  ------------------
  |  Branch (467:25): [True: 939k, False: 873k]
  |  Branch (467:38): [True: 1, False: 939k]
  ------------------
  468|      1|                        throw StaticError(next.location,
  469|      1|                                          "cannot use +: syntax sugar in a method: " + next.data);
  470|      1|                    }
  471|  1.81M|                    if (kind != ObjectField::FIELD_EXPR) {
  ------------------
  |  Branch (471:25): [True: 1.77M, False: 40.2k]
  ------------------
  472|  1.77M|                        if (!literal_fields.insert(next.data).second) {
  ------------------
  |  Branch (472:29): [True: 8, False: 1.77M]
  ------------------
  473|      8|                            throw StaticError(next.location, "duplicate field: " + next.data);
  474|      8|                        }
  475|  1.77M|                    }
  476|       |
  477|  1.81M|                    AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  478|       |
  479|  1.81M|                    Fodder comma_fodder;
  480|  1.81M|                    next = pop();
  481|  1.81M|                    if (next.kind == Token::COMMA) {
  ------------------
  |  Branch (481:25): [True: 1.54M, False: 271k]
  ------------------
  482|  1.54M|                        comma_fodder = next.fodder;
  483|  1.54M|                        next = pop();
  484|  1.54M|                        got_comma = true;
  485|  1.54M|                    }
  486|  1.81M|                    fields.emplace_back(kind,
  487|  1.81M|                                        fodder1,
  488|  1.81M|                                        fodder2,
  489|  1.81M|                                        fodder_l,
  490|  1.81M|                                        fodder_r,
  491|  1.81M|                                        field_hide,
  492|  1.81M|                                        plus_sugar,
  493|  1.81M|                                        is_method,
  494|  1.81M|                                        expr1,
  495|  1.81M|                                        id,
  496|  1.81M|                                        idLocation,
  497|  1.81M|                                        params,
  498|  1.81M|                                        meth_comma,
  499|  1.81M|                                        op.fodder,
  500|  1.81M|                                        body,
  501|  1.81M|                                        nullptr,
  502|  1.81M|                                        comma_fodder);
  503|  1.81M|                } break;
  504|       |
  505|  53.5k|                case Token::LOCAL: {
  ------------------
  |  Branch (505:17): [True: 53.5k, False: 1.85M]
  ------------------
  506|  53.5k|                    Fodder local_fodder = next.fodder;
  507|  53.5k|                    Token var_id = popExpect(Token::IDENTIFIER);
  508|  53.5k|                    auto *id = alloc->makeIdentifier(var_id.data32());
  509|       |
  510|  53.5k|                    if (binds.find(id) != binds.end()) {
  ------------------
  |  Branch (510:25): [True: 4, False: 53.5k]
  ------------------
  511|      4|                        throw StaticError(var_id.location, "duplicate local var: " + var_id.data);
  512|      4|                    }
  513|  53.5k|                    bool is_method = false;
  514|  53.5k|                    bool func_comma = false;
  515|  53.5k|                    ArgParams params;
  516|  53.5k|                    Fodder paren_l_fodder;
  517|  53.5k|                    Fodder paren_r_fodder;
  518|  53.5k|                    if (peek().kind == Token::PAREN_L) {
  ------------------
  |  Branch (518:25): [True: 7.85k, False: 45.7k]
  ------------------
  519|  7.85k|                        Token paren_l = pop();
  520|  7.85k|                        paren_l_fodder = paren_l.fodder;
  521|  7.85k|                        is_method = true;
  522|  7.85k|                        params = parseParams("function parameter", func_comma, paren_r_fodder, current_depth);
  523|  7.85k|                    }
  524|  53.5k|                    Token eq = popExpect(Token::OPERATOR, "=");
  525|  53.5k|                    AST *body = parse(MAX_PRECEDENCE, current_depth + 1);
  526|  53.5k|                    binds.insert(id);
  527|       |
  528|  53.5k|                    Fodder comma_fodder;
  529|  53.5k|                    next = pop();
  530|  53.5k|                    if (next.kind == Token::COMMA) {
  ------------------
  |  Branch (530:25): [True: 52.9k, False: 621]
  ------------------
  531|  52.9k|                        comma_fodder = next.fodder;
  532|  52.9k|                        next = pop();
  533|  52.9k|                        got_comma = true;
  534|  52.9k|                    }
  535|  53.5k|                    fields.push_back(ObjectField::Local(local_fodder,
  536|  53.5k|                                                        var_id.fodder,
  537|  53.5k|                                                        paren_l_fodder,
  538|  53.5k|                                                        paren_r_fodder,
  539|  53.5k|                                                        is_method,
  540|  53.5k|                                                        id,
  541|  53.5k|                                                        params,
  542|  53.5k|                                                        func_comma,
  543|  53.5k|                                                        eq.fodder,
  544|  53.5k|                                                        body,
  545|  53.5k|                                                        comma_fodder));
  546|       |
  547|  53.5k|                } break;
  548|       |
  549|  43.7k|                case Token::ASSERT: {
  ------------------
  |  Branch (549:17): [True: 43.7k, False: 1.86M]
  ------------------
  550|  43.7k|                    Fodder assert_fodder = next.fodder;
  551|  43.7k|                    AST *cond = parse(MAX_PRECEDENCE, current_depth + 1);
  552|  43.7k|                    AST *msg = nullptr;
  553|  43.7k|                    Fodder colon_fodder;
  554|  43.7k|                    if (peek().kind == Token::OPERATOR && peek().data == ":") {
  ------------------
  |  Branch (554:25): [True: 42.6k, False: 1.02k]
  |  Branch (554:59): [True: 42.6k, False: 1]
  ------------------
  555|  42.6k|                        Token colon = pop();
  556|  42.6k|                        colon_fodder = colon.fodder;
  557|  42.6k|                        msg = parse(MAX_PRECEDENCE, current_depth + 1);
  558|  42.6k|                    }
  559|       |
  560|  43.7k|                    Fodder comma_fodder;
  561|  43.7k|                    next = pop();
  562|  43.7k|                    if (next.kind == Token::COMMA) {
  ------------------
  |  Branch (562:25): [True: 41.6k, False: 2.02k]
  ------------------
  563|  41.6k|                        comma_fodder = next.fodder;
  564|  41.6k|                        next = pop();
  565|  41.6k|                        got_comma = true;
  566|  41.6k|                    }
  567|  43.7k|                    fields.push_back(
  568|  43.7k|                        ObjectField::Assert(assert_fodder, cond, colon_fodder, msg, comma_fodder));
  569|  43.7k|                } break;
  570|       |
  571|     78|                default: throw unexpected(next, "parsing field definition");
  ------------------
  |  Branch (571:17): [True: 78, False: 1.91M]
  ------------------
  572|  1.91M|            }
  573|       |
  574|  1.91M|        } while (true);
  ------------------
  |  Branch (574:18): [True: 1.90M, Folded]
  ------------------
  575|   420k|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14spanERKNS0_5TokenES4_:
   89|  14.7M|{
   90|  14.7M|    return LocationRange(begin.location.file, begin.location.begin, end.location.end);
   91|  14.7M|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser23parseComprehensionSpecsENS0_5Token4KindENSt3__16vectorINS0_13FodderElementENS5_9allocatorIS7_EEEERNS6_INS0_17ComprehensionSpecENS8_ISB_EEEEj:
  588|   278k|    {
  589|   278k|        if (current_depth >= MAX_PARSER_DEPTH) {
  ------------------
  |  Branch (589:13): [True: 0, False: 278k]
  ------------------
  590|      0|            throw StaticError(peek().location, "Exceeded maximum parse depth limit.");
  591|      0|        }
  592|       |
  593|   307k|        while (true) {
  ------------------
  |  Branch (593:16): [True: 306k, Folded]
  ------------------
  594|   306k|            LocationRange l;
  595|   306k|            Token id_token = popExpect(Token::IDENTIFIER);
  596|   306k|            const Identifier *id = alloc->makeIdentifier(id_token.data32());
  597|   306k|            Token in_token = popExpect(Token::IN);
  598|   306k|            AST *arr = parse(MAX_PRECEDENCE, current_depth + 1);
  599|   306k|            specs.emplace_back(
  600|   306k|                ComprehensionSpec::FOR, for_fodder, id_token.fodder, id, in_token.fodder, arr);
  601|       |
  602|   306k|            Token maybe_if = pop();
  603|   379k|            for (; maybe_if.kind == Token::IF; maybe_if = pop()) {
  ------------------
  |  Branch (603:20): [True: 73.0k, False: 306k]
  ------------------
  604|  73.0k|                AST *cond = parse(MAX_PRECEDENCE, current_depth + 1);
  605|  73.0k|                specs.emplace_back(
  606|  73.0k|                    ComprehensionSpec::IF, maybe_if.fodder, Fodder{}, nullptr, Fodder{}, cond);
  607|  73.0k|            }
  608|   306k|            if (maybe_if.kind == end) {
  ------------------
  |  Branch (608:17): [True: 277k, False: 28.9k]
  ------------------
  609|   277k|                return maybe_if;
  610|   277k|            }
  611|  28.9k|            if (maybe_if.kind != Token::FOR) {
  ------------------
  |  Branch (611:17): [True: 15, False: 28.9k]
  ------------------
  612|     15|                std::stringstream ss;
  613|     15|                ss << "expected for, if or " << end << " after for clause, got: " << maybe_if;
  614|     15|                throw StaticError(maybe_if.location, ss.str());
  615|     15|            }
  616|  28.9k|            for_fodder = maybe_if.fodder;
  617|  28.9k|        }
  618|   278k|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14spanERKNS0_5TokenE:
   84|  31.9M|{
   85|  31.9M|    return LocationRange(begin.location.file, begin.location.begin, begin.location.end);
   86|  31.9M|}
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_16Parser10parseInfixEPNS0_3ASTERKNS0_5TokenEjj:
 1009|  34.2M|    {
 1010|  34.2M|        if (current_depth >= MAX_PARSER_DEPTH) {
  ------------------
  |  Branch (1010:13): [True: 0, False: 34.2M]
  ------------------
 1011|      0|            throw StaticError(peek().location, "Exceeded maximum parse depth limit.");
 1012|      0|        }
 1013|       |
 1014|  53.7M|        while (true) {
  ------------------
  |  Branch (1014:16): [True: 53.7M, Folded]
  ------------------
 1015|       |
 1016|  53.7M|            BinaryOp bop = BOP_PLUS;
 1017|  53.7M|            unsigned op_precedence = 0;
 1018|       |
 1019|  53.7M|            switch (peek().kind) {
 1020|       |                // Logical / arithmetic binary operator.
 1021|  2.08k|                case Token::IN:
  ------------------
  |  Branch (1021:17): [True: 2.08k, False: 53.7M]
  ------------------
 1022|  8.45M|                case Token::OPERATOR:
  ------------------
  |  Branch (1022:17): [True: 8.45M, False: 45.3M]
  ------------------
 1023|       |                    // These occur if the outer statement was an assert or array slice.
 1024|       |                    // Either way, we terminate the parsing here.
 1025|  8.45M|                    if (peek().data == ":" || peek().data == "::") {
  ------------------
  |  Branch (1025:25): [True: 641k, False: 7.81M]
  |  Branch (1025:47): [True: 362, False: 7.81M]
  ------------------
 1026|   641k|                        return lhs;
 1027|   641k|                    }
 1028|  7.81M|                    if (!op_is_binary(peek().data, bop)) {
  ------------------
  |  Branch (1028:25): [True: 138, False: 7.81M]
  ------------------
 1029|    138|                        std::stringstream ss;
 1030|    138|                        ss << "not a binary operator: " << peek().data;
 1031|    138|                        throw StaticError(peek().location, ss.str());
 1032|    138|                    }
 1033|  7.81M|                    op_precedence = precedence_map[bop];
 1034|  7.81M|                    break;
 1035|       |
 1036|       |                // Index, Apply
 1037|  5.43M|                case Token::DOT:
  ------------------
  |  Branch (1037:17): [True: 5.43M, False: 48.3M]
  ------------------
 1038|  6.75M|                case Token::BRACKET_L:
  ------------------
  |  Branch (1038:17): [True: 1.32M, False: 52.4M]
  ------------------
 1039|  12.8M|                case Token::PAREN_L:
  ------------------
  |  Branch (1039:17): [True: 6.13M, False: 47.6M]
  ------------------
 1040|  12.9M|                case Token::BRACE_L:
  ------------------
  |  Branch (1040:17): [True: 58.7k, False: 53.7M]
  ------------------
 1041|  12.9M|                    op_precedence = APPLY_PRECEDENCE;
 1042|  12.9M|                    break;
 1043|       |
 1044|  32.3M|                default:
  ------------------
  |  Branch (1044:17): [True: 32.3M, False: 21.4M]
  ------------------
 1045|       |                    // This happens when we reach EOF or the terminating token of an outer context.
 1046|  32.3M|                    return lhs;
 1047|  53.7M|            }
 1048|       |
 1049|       |            // If higher precedence than the outer recursive call, let that handle it.
 1050|  20.7M|            if (op_precedence >= max_precedence)
  ------------------
  |  Branch (1050:17): [True: 1.25M, False: 19.5M]
  ------------------
 1051|  1.25M|                return lhs;
 1052|       |
 1053|  19.5M|            Token op = pop();
 1054|       |
 1055|  19.5M|            switch (op.kind) {
 1056|  1.32M|                case Token::BRACKET_L: {
  ------------------
  |  Branch (1056:17): [True: 1.32M, False: 18.1M]
  ------------------
 1057|  1.32M|                    bool is_slice;
 1058|  1.32M|                    AST *first = nullptr;
 1059|  1.32M|                    Fodder second_fodder;
 1060|  1.32M|                    AST *second = nullptr;
 1061|  1.32M|                    Fodder third_fodder;
 1062|  1.32M|                    AST *third = nullptr;
 1063|       |
 1064|  1.32M|                    if (peek().kind == Token::BRACKET_R)
  ------------------
  |  Branch (1064:25): [True: 4, False: 1.32M]
  ------------------
 1065|      4|                        throw unexpected(pop(), "parsing index");
 1066|       |
 1067|  1.32M|                    if (peek().data != ":" && peek().data != "::") {
  ------------------
  |  Branch (1067:25): [True: 1.30M, False: 19.2k]
  |  Branch (1067:47): [True: 1.30M, False: 1.19k]
  ------------------
 1068|  1.30M|                        first = parse(MAX_PRECEDENCE, current_depth + 1);
 1069|  1.30M|                    }
 1070|       |
 1071|  1.32M|                    if (peek().kind == Token::OPERATOR && peek().data == "::") {
  ------------------
  |  Branch (1071:25): [True: 141k, False: 1.18M]
  |  Branch (1071:59): [True: 1.19k, False: 140k]
  ------------------
 1072|       |                        // Handle ::
 1073|  1.19k|                        is_slice = true;
 1074|  1.19k|                        Token joined = pop();
 1075|  1.19k|                        second_fodder = joined.fodder;
 1076|       |
 1077|  1.19k|                        if (peek().kind != Token::BRACKET_R)
  ------------------
  |  Branch (1077:29): [True: 841, False: 353]
  ------------------
 1078|    841|                            third = parse(MAX_PRECEDENCE, current_depth + 1);
 1079|       |
 1080|  1.32M|                    } else if (peek().kind != Token::BRACKET_R) {
  ------------------
  |  Branch (1080:32): [True: 140k, False: 1.18M]
  ------------------
 1081|   140k|                        is_slice = true;
 1082|   140k|                        Token delim = pop();
 1083|   140k|                        if (delim.data != ":")
  ------------------
  |  Branch (1083:29): [True: 121, False: 140k]
  ------------------
 1084|    121|                            throw unexpected(delim, "parsing slice");
 1085|       |
 1086|   140k|                        second_fodder = delim.fodder;
 1087|       |
 1088|   140k|                        if (peek().data != ":" && peek().kind != Token::BRACKET_R)
  ------------------
  |  Branch (1088:29): [True: 140k, False: 190]
  |  Branch (1088:51): [True: 62.7k, False: 77.4k]
  ------------------
 1089|  62.7k|                            second = parse(MAX_PRECEDENCE, current_depth + 1);
 1090|       |
 1091|   140k|                        if (peek().kind != Token::BRACKET_R) {
  ------------------
  |  Branch (1091:29): [True: 8.73k, False: 131k]
  ------------------
 1092|  8.73k|                            Token delim = pop();
 1093|  8.73k|                            if (delim.data != ":")
  ------------------
  |  Branch (1093:33): [True: 29, False: 8.70k]
  ------------------
 1094|     29|                                throw unexpected(delim, "parsing slice");
 1095|       |
 1096|  8.70k|                            third_fodder = delim.fodder;
 1097|       |
 1098|  8.70k|                            if (peek().kind != Token::BRACKET_R)
  ------------------
  |  Branch (1098:33): [True: 8.16k, False: 539]
  ------------------
 1099|  8.16k|                                third = parse(MAX_PRECEDENCE, current_depth + 1);
 1100|  8.70k|                        }
 1101|  1.18M|                    } else {
 1102|  1.18M|                        is_slice = false;
 1103|  1.18M|                    }
 1104|  1.32M|                    Token end = popExpect(Token::BRACKET_R);
 1105|  1.32M|                    lhs = alloc->make<Index>(span(begin, end),
 1106|  1.32M|                                             EMPTY_FODDER,
 1107|  1.32M|                                             lhs,
 1108|  1.32M|                                             op.fodder,
 1109|  1.32M|                                             is_slice,
 1110|  1.32M|                                             first,
 1111|  1.32M|                                             second_fodder,
 1112|  1.32M|                                             second,
 1113|  1.32M|                                             third_fodder,
 1114|  1.32M|                                             third,
 1115|  1.32M|                                             end.fodder);
 1116|  1.32M|                    break;
 1117|  1.32M|                }
 1118|  5.43M|                case Token::DOT: {
  ------------------
  |  Branch (1118:17): [True: 5.43M, False: 14.0M]
  ------------------
 1119|  5.43M|                    Token field_id = popExpect(Token::IDENTIFIER);
 1120|  5.43M|                    const Identifier *id = alloc->makeIdentifier(field_id.data32());
 1121|  5.43M|                    lhs = alloc->make<Index>(span(begin, field_id),
 1122|  5.43M|                                             EMPTY_FODDER,
 1123|  5.43M|                                             lhs,
 1124|  5.43M|                                             op.fodder,
 1125|  5.43M|                                             field_id.fodder,
 1126|  5.43M|                                             id);
 1127|  5.43M|                    break;
 1128|  1.32M|                }
 1129|  6.13M|                case Token::PAREN_L: {
  ------------------
  |  Branch (1129:17): [True: 6.13M, False: 13.3M]
  ------------------
 1130|  6.13M|                    ArgParams args;
 1131|  6.13M|                    bool got_comma;
 1132|  6.13M|                    Token end = parseArgs(args, "function argument", got_comma, current_depth);
 1133|  6.13M|                    bool got_named = false;
 1134|  10.2M|                    for (const auto& arg : args) {
  ------------------
  |  Branch (1134:42): [True: 10.2M, False: 6.13M]
  ------------------
 1135|  10.2M|                        if (arg.id != nullptr) {
  ------------------
  |  Branch (1135:29): [True: 26.1k, False: 10.1M]
  ------------------
 1136|  26.1k|                            got_named = true;
 1137|  10.1M|                        } else {
 1138|  10.1M|                            if (got_named) {
  ------------------
  |  Branch (1138:33): [True: 5, False: 10.1M]
  ------------------
 1139|      5|                                throw StaticError(arg.expr->location, "Positional argument after a named argument is not allowed");
 1140|      5|                            }
 1141|  10.1M|                        }
 1142|  10.2M|                    }
 1143|  6.13M|                    bool tailstrict = false;
 1144|  6.13M|                    Fodder tailstrict_fodder;
 1145|  6.13M|                    if (peek().kind == Token::TAILSTRICT) {
  ------------------
  |  Branch (1145:25): [True: 302k, False: 5.83M]
  ------------------
 1146|   302k|                        Token tailstrict_token = pop();
 1147|   302k|                        tailstrict_fodder = tailstrict_token.fodder;
 1148|   302k|                        tailstrict = true;
 1149|   302k|                    }
 1150|  6.13M|                    lhs = alloc->make<Apply>(span(begin, end),
 1151|  6.13M|                                             EMPTY_FODDER,
 1152|  6.13M|                                             lhs,
 1153|  6.13M|                                             op.fodder,
 1154|  6.13M|                                             args,
 1155|  6.13M|                                             got_comma,
 1156|  6.13M|                                             end.fodder,
 1157|  6.13M|                                             tailstrict_fodder,
 1158|  6.13M|                                             tailstrict);
 1159|  6.13M|                    break;
 1160|  6.13M|                }
 1161|  58.7k|                case Token::BRACE_L: {
  ------------------
  |  Branch (1161:17): [True: 58.7k, False: 19.4M]
  ------------------
 1162|  58.7k|                    AST *obj;
 1163|  58.7k|                    Token end = parseObjectRemainder(obj, op, current_depth + 1);
 1164|  58.7k|                    lhs = alloc->make<ApplyBrace>(span(begin, end), EMPTY_FODDER, lhs, obj);
 1165|  58.7k|                    break;
 1166|  6.13M|                }
 1167|       |
 1168|  1.57k|                case Token::IN: {
  ------------------
  |  Branch (1168:17): [True: 1.57k, False: 19.5M]
  ------------------
 1169|  1.57k|                    if (peek().kind == Token::SUPER) {
  ------------------
  |  Branch (1169:25): [True: 115, False: 1.46k]
  ------------------
 1170|    115|                        Token super = pop();
 1171|    115|                        lhs = alloc->make<InSuper>(
 1172|    115|                            span(begin, super), EMPTY_FODDER, lhs, op.fodder, super.fodder);
 1173|  1.46k|                    } else {
 1174|  1.46k|                        AST *rhs = parse(op_precedence, current_depth + 1);
 1175|  1.46k|                        lhs = alloc->make<Binary>(
 1176|  1.46k|                            span(begin, rhs), EMPTY_FODDER, lhs, op.fodder, bop, rhs);
 1177|  1.46k|                    }
 1178|  1.57k|                    break;
 1179|  6.13M|                }
 1180|       |
 1181|  6.56M|                case Token::OPERATOR: {
  ------------------
  |  Branch (1181:17): [True: 6.56M, False: 12.9M]
  ------------------
 1182|  6.56M|                    AST *rhs = parse(op_precedence, current_depth + 1);
 1183|  6.56M|                    lhs = alloc->make<Binary>(
 1184|  6.56M|                        span(begin, rhs), EMPTY_FODDER, lhs, op.fodder, bop, rhs);
 1185|  6.56M|                    break;
 1186|  6.13M|                }
 1187|       |
 1188|      0|                default: {
  ------------------
  |  Branch (1188:17): [True: 0, False: 19.5M]
  ------------------
 1189|      0|                    std::cerr << "Should not be here." << std::endl;
 1190|      0|                    abort();
 1191|  6.13M|                }
 1192|  19.5M|            }
 1193|  19.5M|        }
 1194|       |
 1195|       |        // (1 & ((1 + (1 * 1)) + 1)) & 1
 1196|       |        //
 1197|       |        //
 1198|       |
 1199|       |/*
 1200|       |        // Allocate this on the heap to control stack growth.
 1201|       |        std::unique_ptr<Token> begin_(new Token(peek()));
 1202|       |        const Token &begin = *begin_;
 1203|       |*/
 1204|  34.2M|    }
parser.cpp:_ZN7jsonnet8internal12_GLOBAL__N_112op_is_binaryERKNSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEERNS0_8BinaryOpE:
   75|  7.81M|{
   76|  7.81M|    auto it = binary_map.find(op);
   77|  7.81M|    if (it == binary_map.end())
  ------------------
  |  Branch (77:9): [True: 138, False: 7.81M]
  ------------------
   78|    138|        return false;
   79|  7.81M|    bop = it->second;
   80|  7.81M|    return true;
   81|  7.81M|}

_ZN7jsonnet8internal12CompilerPass6fodderERNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEE:
   22|  29.0M|{
   23|  29.0M|    for (auto &f : fodder)
  ------------------
  |  Branch (23:18): [True: 83.6k, False: 29.0M]
  ------------------
   24|  83.6k|        fodderElement(f);
   25|  29.0M|}
_ZN7jsonnet8internal12CompilerPass6paramsERNSt3__16vectorINS0_13FodderElementENS2_9allocatorIS4_EEEERNS3_INS0_8ArgParamENS5_IS9_EEEES8_:
   43|   655k|{
   44|   655k|    fodder(fodder_l);
   45|  1.22M|    for (auto &param : params) {
  ------------------
  |  Branch (45:22): [True: 1.22M, False: 655k]
  ------------------
   46|  1.22M|        fodder(param.idFodder);
   47|  1.22M|        if (param.expr) {
  ------------------
  |  Branch (47:13): [True: 1.12M, False: 94.6k]
  ------------------
   48|  1.12M|            fodder(param.eqFodder);
   49|  1.12M|            expr(param.expr);
   50|  1.12M|        }
   51|  1.22M|        fodder(param.commaFodder);
   52|  1.22M|    }
   53|   655k|    fodder(fodder_r);
   54|   655k|}
_ZN7jsonnet8internal12CompilerPass4exprERPNS0_3ASTE:
  110|  14.8M|{
  111|  14.8M|    fodder(ast_->openFodder);
  112|  14.8M|    visitExpr(ast_);
  113|  14.8M|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_5ApplyE:
  116|   607k|{
  117|   607k|    expr(ast->target);
  118|   607k|    params(ast->fodderL, ast->args, ast->fodderR);
  119|   607k|    if (ast->tailstrict) {
  ------------------
  |  Branch (119:9): [True: 315k, False: 292k]
  ------------------
  120|   315k|        fodder(ast->tailstrictFodder);
  121|   315k|    }
  122|   607k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_5ArrayE:
  131|   607k|{
  132|   607k|    for (auto &element : ast->elements) {
  ------------------
  |  Branch (132:24): [True: 563k, False: 607k]
  ------------------
  133|   563k|        expr(element.expr);
  134|   563k|        fodder(element.commaFodder);
  135|   563k|    }
  136|   607k|    fodder(ast->closeFodder);
  137|   607k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_6BinaryE:
  159|  1.06M|{
  160|  1.06M|    expr(ast->left);
  161|  1.06M|    fodder(ast->opFodder);
  162|  1.06M|    expr(ast->right);
  163|  1.06M|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_11ConditionalE:
  166|   542k|{
  167|   542k|    expr(ast->cond);
  168|   542k|    fodder(ast->thenFodder);
  169|   542k|    if (ast->branchFalse != nullptr) {
  ------------------
  |  Branch (169:9): [True: 542k, False: 0]
  ------------------
  170|   542k|        expr(ast->branchTrue);
  171|   542k|        fodder(ast->elseFodder);
  172|   542k|        expr(ast->branchFalse);
  173|   542k|    } else {
  174|      0|        expr(ast->branchTrue);
  175|      0|    }
  176|   542k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_5ErrorE:
  179|   138k|{
  180|   138k|    expr(ast->expr);
  181|   138k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_8FunctionE:
  184|  47.6k|{
  185|  47.6k|    params(ast->parenLeftFodder, ast->params, ast->parenRightFodder);
  186|  47.6k|    expr(ast->body);
  187|  47.6k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_6ImportE:
  190|    844|{
  191|    844|    visit(ast->file);
  192|    844|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_9ImportstrE:
  195|  1.70k|{
  196|  1.70k|    visit(ast->file);
  197|  1.70k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_9ImportbinE:
  200|  30.6k|{
  201|  30.6k|    visit(ast->file);
  202|  30.6k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_7InSuperE:
  205|   140k|{
  206|   140k|    expr(ast->element);
  207|   140k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_5IndexE:
  210|   714k|{
  211|   714k|    expr(ast->target);
  212|   714k|    if (ast->id != nullptr) {
  ------------------
  |  Branch (212:9): [True: 0, False: 714k]
  ------------------
  213|   714k|    } else {
  214|   714k|        if (ast->isSlice) {
  ------------------
  |  Branch (214:13): [True: 0, False: 714k]
  ------------------
  215|      0|            if (ast->index != nullptr)
  ------------------
  |  Branch (215:17): [True: 0, False: 0]
  ------------------
  216|      0|                expr(ast->index);
  217|      0|            if (ast->end != nullptr)
  ------------------
  |  Branch (217:17): [True: 0, False: 0]
  ------------------
  218|      0|                expr(ast->end);
  219|      0|            if (ast->step != nullptr)
  ------------------
  |  Branch (219:17): [True: 0, False: 0]
  ------------------
  220|      0|                expr(ast->step);
  221|   714k|        } else {
  222|   714k|            expr(ast->index);
  223|   714k|        }
  224|   714k|    }
  225|   714k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_5LocalE:
  228|   326k|{
  229|   326k|    assert(ast->binds.size() > 0);
  ------------------
  |  Branch (229:5): [True: 326k, False: 0]
  ------------------
  230|  1.88M|    for (auto &bind : ast->binds) {
  ------------------
  |  Branch (230:21): [True: 1.88M, False: 326k]
  ------------------
  231|  1.88M|        fodder(bind.varFodder);
  232|  1.88M|        if (bind.functionSugar) {
  ------------------
  |  Branch (232:13): [True: 0, False: 1.88M]
  ------------------
  233|      0|            params(bind.parenLeftFodder, bind.params, bind.parenRightFodder);
  234|      0|        }
  235|  1.88M|        fodder(bind.opFodder);
  236|  1.88M|        expr(bind.body);
  237|  1.88M|        fodder(bind.closeFodder);
  238|  1.88M|    }
  239|   326k|    expr(ast->body);
  240|   326k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_15DesugaredObjectE:
  249|   581k|{
  250|   581k|    for (AST *assert : ast->asserts) {
  ------------------
  |  Branch (250:22): [True: 90.6k, False: 581k]
  ------------------
  251|  90.6k|        expr(assert);
  252|  90.6k|    }
  253|   610k|    for (auto &field : ast->fields) {
  ------------------
  |  Branch (253:22): [True: 610k, False: 581k]
  ------------------
  254|   610k|        expr(field.name);
  255|   610k|        expr(field.body);
  256|   610k|    }
  257|   581k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_25ObjectComprehensionSimpleE:
  267|  47.1k|{
  268|  47.1k|    expr(ast->field);
  269|  47.1k|    expr(ast->value);
  270|  47.1k|    expr(ast->array);
  271|  47.1k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_10SuperIndexE:
  280|   148k|{
  281|   148k|    if (ast->id != nullptr) {
  ------------------
  |  Branch (281:9): [True: 0, False: 148k]
  ------------------
  282|   148k|    } else {
  283|   148k|        expr(ast->index);
  284|   148k|    }
  285|   148k|}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_5UnaryE:
  288|  3.23M|{
  289|  3.23M|    expr(ast->expr);
  290|  3.23M|}
_ZN7jsonnet8internal12CompilerPass9visitExprERPNS0_3ASTE:
  300|  14.8M|{
  301|  14.8M|    switch(ast_->type) {
  302|   607k|        VISIT(ast_, AST_APPLY, Apply);
  ------------------
  |  |  293|   607k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 607k, False: 14.2M]
  |  |  ------------------
  |  |  294|   607k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   607k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   607k|     visit(ast); \
  |  |  297|   607k|   } break
  ------------------
  |  Branch (302:9): [True: 607k, False: 0]
  ------------------
  303|      0|        VISIT(ast_, AST_APPLY_BRACE, ApplyBrace);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 14.8M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (303:9): [True: 0, False: 0]
  ------------------
  304|   607k|        VISIT(ast_, AST_ARRAY, Array);
  ------------------
  |  |  293|   607k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 607k, False: 14.2M]
  |  |  ------------------
  |  |  294|   607k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   607k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   607k|     visit(ast); \
  |  |  297|   607k|   } break
  ------------------
  |  Branch (304:9): [True: 607k, False: 0]
  ------------------
  305|      0|        VISIT(ast_, AST_ARRAY_COMPREHENSION, ArrayComprehension);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 14.8M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (305:9): [True: 0, False: 0]
  ------------------
  306|       |        // VISIT(ast_, AST_ARRAY_COMPREHENSION, ArrayComprehensionSimple);
  307|      0|        VISIT(ast_, AST_ASSERT, Assert);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 14.8M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (307:9): [True: 0, False: 0]
  ------------------
  308|  1.06M|        VISIT(ast_, AST_BINARY, Binary);
  ------------------
  |  |  293|  1.06M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 1.06M, False: 13.8M]
  |  |  ------------------
  |  |  294|  1.06M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  1.06M|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  1.06M|     visit(ast); \
  |  |  297|  1.06M|   } break
  ------------------
  |  Branch (308:9): [True: 1.06M, False: 0]
  ------------------
  309|      0|        VISIT(ast_, AST_BUILTIN_FUNCTION, BuiltinFunction);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 14.8M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (309:9): [True: 0, False: 0]
  ------------------
  310|      0|        VISIT(ast_, AST_BUILTIN_FUNCTION_BODY, BuiltinFunctionBody);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 14.8M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (310:9): [True: 0, False: 0]
  ------------------
  311|   542k|        VISIT(ast_, AST_CONDITIONAL, Conditional);
  ------------------
  |  |  293|   542k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 542k, False: 14.3M]
  |  |  ------------------
  |  |  294|   542k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   542k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   542k|     visit(ast); \
  |  |  297|   542k|   } break
  ------------------
  |  Branch (311:9): [True: 542k, False: 0]
  ------------------
  312|   581k|        VISIT(ast_, AST_DESUGARED_OBJECT, DesugaredObject);
  ------------------
  |  |  293|   581k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 581k, False: 14.3M]
  |  |  ------------------
  |  |  294|   581k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   581k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   581k|     visit(ast); \
  |  |  297|   581k|   } break
  ------------------
  |  Branch (312:9): [True: 581k, False: 0]
  ------------------
  313|      0|        VISIT(ast_, AST_DOLLAR, Dollar);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 14.8M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (313:9): [True: 0, False: 0]
  ------------------
  314|   138k|        VISIT(ast_, AST_ERROR, Error);
  ------------------
  |  |  293|   138k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 138k, False: 14.7M]
  |  |  ------------------
  |  |  294|   138k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   138k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   138k|     visit(ast); \
  |  |  297|   138k|   } break
  ------------------
  |  Branch (314:9): [True: 138k, False: 0]
  ------------------
  315|  47.6k|        VISIT(ast_, AST_FUNCTION, Function);
  ------------------
  |  |  293|  47.6k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 47.6k, False: 14.8M]
  |  |  ------------------
  |  |  294|  47.6k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  47.6k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  47.6k|     visit(ast); \
  |  |  297|  47.6k|   } break
  ------------------
  |  Branch (315:9): [True: 47.6k, False: 0]
  ------------------
  316|    844|        VISIT(ast_, AST_IMPORT, Import);
  ------------------
  |  |  293|    844|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 844, False: 14.8M]
  |  |  ------------------
  |  |  294|    844|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|    844|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|    844|     visit(ast); \
  |  |  297|    844|   } break
  ------------------
  |  Branch (316:9): [True: 844, False: 0]
  ------------------
  317|  1.70k|        VISIT(ast_, AST_IMPORTSTR, Importstr);
  ------------------
  |  |  293|  1.70k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 1.70k, False: 14.8M]
  |  |  ------------------
  |  |  294|  1.70k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  1.70k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  1.70k|     visit(ast); \
  |  |  297|  1.70k|   } break
  ------------------
  |  Branch (317:9): [True: 1.70k, False: 0]
  ------------------
  318|  30.6k|        VISIT(ast_, AST_IMPORTBIN, Importbin);
  ------------------
  |  |  293|  30.6k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 30.6k, False: 14.8M]
  |  |  ------------------
  |  |  294|  30.6k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  30.6k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  30.6k|     visit(ast); \
  |  |  297|  30.6k|   } break
  ------------------
  |  Branch (318:9): [True: 30.6k, False: 0]
  ------------------
  319|   714k|        VISIT(ast_, AST_INDEX, Index);
  ------------------
  |  |  293|   714k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 714k, False: 14.1M]
  |  |  ------------------
  |  |  294|   714k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   714k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   714k|     visit(ast); \
  |  |  297|   714k|   } break
  ------------------
  |  Branch (319:9): [True: 714k, False: 0]
  ------------------
  320|   140k|        VISIT(ast_, AST_IN_SUPER, InSuper);
  ------------------
  |  |  293|   140k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 140k, False: 14.7M]
  |  |  ------------------
  |  |  294|   140k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   140k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   140k|     visit(ast); \
  |  |  297|   140k|   } break
  ------------------
  |  Branch (320:9): [True: 140k, False: 0]
  ------------------
  321|  93.3k|        VISIT(ast_, AST_LITERAL_BOOLEAN, LiteralBoolean);
  ------------------
  |  |  293|  93.3k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 93.3k, False: 14.7M]
  |  |  ------------------
  |  |  294|  93.3k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  93.3k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  93.3k|     visit(ast); \
  |  |  297|  93.3k|   } break
  ------------------
  |  Branch (321:9): [True: 93.3k, False: 0]
  ------------------
  322|   107k|        VISIT(ast_, AST_LITERAL_NULL, LiteralNull);
  ------------------
  |  |  293|   107k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 107k, False: 14.7M]
  |  |  ------------------
  |  |  294|   107k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   107k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   107k|     visit(ast); \
  |  |  297|   107k|   } break
  ------------------
  |  Branch (322:9): [True: 107k, False: 0]
  ------------------
  323|  1.12M|        VISIT(ast_, AST_LITERAL_NUMBER, LiteralNumber);
  ------------------
  |  |  293|  1.12M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 1.12M, False: 13.7M]
  |  |  ------------------
  |  |  294|  1.12M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  1.12M|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  1.12M|     visit(ast); \
  |  |  297|  1.12M|   } break
  ------------------
  |  Branch (323:9): [True: 1.12M, False: 0]
  ------------------
  324|  1.71M|        VISIT(ast_, AST_LITERAL_STRING, LiteralString);
  ------------------
  |  |  293|  1.71M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 1.71M, False: 13.1M]
  |  |  ------------------
  |  |  294|  1.71M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  1.71M|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  1.71M|     visit(ast); \
  |  |  297|  1.71M|   } break
  ------------------
  |  Branch (324:9): [True: 1.71M, False: 0]
  ------------------
  325|   326k|        VISIT(ast_, AST_LOCAL, Local);
  ------------------
  |  |  293|   326k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 326k, False: 14.5M]
  |  |  ------------------
  |  |  294|   326k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   326k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   326k|     visit(ast); \
  |  |  297|   326k|   } break
  ------------------
  |  Branch (325:9): [True: 326k, False: 0]
  ------------------
  326|      0|        VISIT(ast_, AST_OBJECT, Object);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 14.8M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (326:9): [True: 0, False: 0]
  ------------------
  327|      0|        VISIT(ast_, AST_OBJECT_COMPREHENSION, ObjectComprehension);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 14.8M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (327:9): [True: 0, False: 0]
  ------------------
  328|  47.1k|        VISIT(ast_, AST_OBJECT_COMPREHENSION_SIMPLE, ObjectComprehensionSimple);
  ------------------
  |  |  293|  47.1k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 47.1k, False: 14.8M]
  |  |  ------------------
  |  |  294|  47.1k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  47.1k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  47.1k|     visit(ast); \
  |  |  297|  47.1k|   } break
  ------------------
  |  Branch (328:9): [True: 47.1k, False: 0]
  ------------------
  329|      0|        VISIT(ast_, AST_PARENS, Parens);
  ------------------
  |  |  293|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 0, False: 14.8M]
  |  |  ------------------
  |  |  294|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|      0|     visit(ast); \
  |  |  297|      0|   } break
  ------------------
  |  Branch (329:9): [True: 0, False: 0]
  ------------------
  330|  15.3k|        VISIT(ast_, AST_SELF, Self);
  ------------------
  |  |  293|  15.3k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 15.3k, False: 14.8M]
  |  |  ------------------
  |  |  294|  15.3k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  15.3k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  15.3k|     visit(ast); \
  |  |  297|  15.3k|   } break
  ------------------
  |  Branch (330:9): [True: 15.3k, False: 0]
  ------------------
  331|   148k|        VISIT(ast_, AST_SUPER_INDEX, SuperIndex);
  ------------------
  |  |  293|   148k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 148k, False: 14.7M]
  |  |  ------------------
  |  |  294|   148k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|   148k|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|   148k|     visit(ast); \
  |  |  297|   148k|   } break
  ------------------
  |  Branch (331:9): [True: 148k, False: 0]
  ------------------
  332|  3.23M|        VISIT(ast_, AST_UNARY, Unary);
  ------------------
  |  |  293|  3.23M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 3.23M, False: 11.6M]
  |  |  ------------------
  |  |  294|  3.23M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  3.23M|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  3.23M|     visit(ast); \
  |  |  297|  3.23M|   } break
  ------------------
  |  Branch (332:9): [True: 3.23M, False: 0]
  ------------------
  333|  3.60M|        VISIT(ast_, AST_VAR, Var);
  ------------------
  |  |  293|  3.60M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (293:4): [True: 3.60M, False: 11.2M]
  |  |  ------------------
  |  |  294|  3.60M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  295|  3.60M|     auto *ast = static_cast<astClass *>(var); \
  |  |  296|  3.60M|     visit(ast); \
  |  |  297|  3.60M|   } break
  ------------------
  |  Branch (333:9): [True: 3.60M, False: 0]
  ------------------
  334|      0|        default:
  ------------------
  |  Branch (334:9): [True: 0, False: 14.8M]
  ------------------
  335|      0|            std::cerr << "INTERNAL ERROR: Unknown AST: " << ast_ << std::endl;
  336|      0|            std::abort();
  337|      0|            break;
  338|  14.8M|    }
  339|  14.8M|}
_ZN7jsonnet8internal9ClonePass4exprERPNS0_3ASTE:
  362|  13.5M|{
  363|  13.5M|    switch(ast_->type) {
  364|   560k|        CLONE(ast_, AST_APPLY, Apply);
  ------------------
  |  |  355|   560k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 560k, False: 13.0M]
  |  |  ------------------
  |  |  356|   560k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   560k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   560k|     var = alloc.clone(ast); \
  |  |  359|   560k|   } break
  ------------------
  |  Branch (364:9): [True: 560k, False: 0]
  ------------------
  365|      0|        CLONE(ast_, AST_APPLY_BRACE, ApplyBrace);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 13.5M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (365:9): [True: 0, False: 0]
  ------------------
  366|   563k|        CLONE(ast_, AST_ARRAY, Array);
  ------------------
  |  |  355|   563k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 563k, False: 13.0M]
  |  |  ------------------
  |  |  356|   563k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   563k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   563k|     var = alloc.clone(ast); \
  |  |  359|   563k|   } break
  ------------------
  |  Branch (366:9): [True: 563k, False: 0]
  ------------------
  367|      0|        CLONE(ast_, AST_ARRAY_COMPREHENSION, ArrayComprehension);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 13.5M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (367:9): [True: 0, False: 0]
  ------------------
  368|       |        // CLONE(ast_, AST_ARRAY_COMPREHENSION, ArrayComprehensionSimple);
  369|      0|        CLONE(ast_, AST_ASSERT, Assert);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 13.5M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (369:9): [True: 0, False: 0]
  ------------------
  370|   971k|        CLONE(ast_, AST_BINARY, Binary);
  ------------------
  |  |  355|   971k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 971k, False: 12.6M]
  |  |  ------------------
  |  |  356|   971k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   971k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   971k|     var = alloc.clone(ast); \
  |  |  359|   971k|   } break
  ------------------
  |  Branch (370:9): [True: 971k, False: 0]
  ------------------
  371|      0|        CLONE(ast_, AST_BUILTIN_FUNCTION, BuiltinFunction);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 13.5M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (371:9): [True: 0, False: 0]
  ------------------
  372|   502k|        CLONE(ast_, AST_CONDITIONAL, Conditional);
  ------------------
  |  |  355|   502k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 502k, False: 13.0M]
  |  |  ------------------
  |  |  356|   502k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   502k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   502k|     var = alloc.clone(ast); \
  |  |  359|   502k|   } break
  ------------------
  |  Branch (372:9): [True: 502k, False: 0]
  ------------------
  373|   530k|        CLONE(ast_, AST_DESUGARED_OBJECT, DesugaredObject);
  ------------------
  |  |  355|   530k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 530k, False: 13.0M]
  |  |  ------------------
  |  |  356|   530k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   530k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   530k|     var = alloc.clone(ast); \
  |  |  359|   530k|   } break
  ------------------
  |  Branch (373:9): [True: 530k, False: 0]
  ------------------
  374|      0|        CLONE(ast_, AST_DOLLAR, Dollar);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 13.5M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (374:9): [True: 0, False: 0]
  ------------------
  375|   127k|        CLONE(ast_, AST_ERROR, Error);
  ------------------
  |  |  355|   127k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 127k, False: 13.4M]
  |  |  ------------------
  |  |  356|   127k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   127k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   127k|     var = alloc.clone(ast); \
  |  |  359|   127k|   } break
  ------------------
  |  Branch (375:9): [True: 127k, False: 0]
  ------------------
  376|  45.4k|        CLONE(ast_, AST_FUNCTION, Function);
  ------------------
  |  |  355|  45.4k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 45.4k, False: 13.5M]
  |  |  ------------------
  |  |  356|  45.4k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  45.4k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  45.4k|     var = alloc.clone(ast); \
  |  |  359|  45.4k|   } break
  ------------------
  |  Branch (376:9): [True: 45.4k, False: 0]
  ------------------
  377|    814|        CLONE(ast_, AST_IMPORT, Import);
  ------------------
  |  |  355|    814|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 814, False: 13.5M]
  |  |  ------------------
  |  |  356|    814|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|    814|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|    814|     var = alloc.clone(ast); \
  |  |  359|    814|   } break
  ------------------
  |  Branch (377:9): [True: 814, False: 0]
  ------------------
  378|  1.24k|        CLONE(ast_, AST_IMPORTSTR, Importstr);
  ------------------
  |  |  355|  1.24k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 1.24k, False: 13.5M]
  |  |  ------------------
  |  |  356|  1.24k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  1.24k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  1.24k|     var = alloc.clone(ast); \
  |  |  359|  1.24k|   } break
  ------------------
  |  Branch (378:9): [True: 1.24k, False: 0]
  ------------------
  379|  28.0k|        CLONE(ast_, AST_IMPORTBIN, Importbin);
  ------------------
  |  |  355|  28.0k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 28.0k, False: 13.5M]
  |  |  ------------------
  |  |  356|  28.0k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  28.0k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  28.0k|     var = alloc.clone(ast); \
  |  |  359|  28.0k|   } break
  ------------------
  |  Branch (379:9): [True: 28.0k, False: 0]
  ------------------
  380|   657k|        CLONE(ast_, AST_INDEX, Index);
  ------------------
  |  |  355|   657k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 657k, False: 12.9M]
  |  |  ------------------
  |  |  356|   657k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   657k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   657k|     var = alloc.clone(ast); \
  |  |  359|   657k|   } break
  ------------------
  |  Branch (380:9): [True: 657k, False: 0]
  ------------------
  381|   140k|        CLONE(ast_, AST_IN_SUPER, InSuper);
  ------------------
  |  |  355|   140k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 140k, False: 13.4M]
  |  |  ------------------
  |  |  356|   140k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   140k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   140k|     var = alloc.clone(ast); \
  |  |  359|   140k|   } break
  ------------------
  |  Branch (381:9): [True: 140k, False: 0]
  ------------------
  382|  84.6k|        CLONE(ast_, AST_LITERAL_BOOLEAN, LiteralBoolean);
  ------------------
  |  |  355|  84.6k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 84.6k, False: 13.5M]
  |  |  ------------------
  |  |  356|  84.6k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  84.6k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  84.6k|     var = alloc.clone(ast); \
  |  |  359|  84.6k|   } break
  ------------------
  |  Branch (382:9): [True: 84.6k, False: 0]
  ------------------
  383|   101k|        CLONE(ast_, AST_LITERAL_NULL, LiteralNull);
  ------------------
  |  |  355|   101k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 101k, False: 13.4M]
  |  |  ------------------
  |  |  356|   101k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   101k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   101k|     var = alloc.clone(ast); \
  |  |  359|   101k|   } break
  ------------------
  |  Branch (383:9): [True: 101k, False: 0]
  ------------------
  384|  1.02M|        CLONE(ast_, AST_LITERAL_NUMBER, LiteralNumber);
  ------------------
  |  |  355|  1.02M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 1.02M, False: 12.5M]
  |  |  ------------------
  |  |  356|  1.02M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  1.02M|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  1.02M|     var = alloc.clone(ast); \
  |  |  359|  1.02M|   } break
  ------------------
  |  Branch (384:9): [True: 1.02M, False: 0]
  ------------------
  385|  1.58M|        CLONE(ast_, AST_LITERAL_STRING, LiteralString);
  ------------------
  |  |  355|  1.58M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 1.58M, False: 12.0M]
  |  |  ------------------
  |  |  356|  1.58M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  1.58M|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  1.58M|     var = alloc.clone(ast); \
  |  |  359|  1.58M|   } break
  ------------------
  |  Branch (385:9): [True: 1.58M, False: 0]
  ------------------
  386|   314k|        CLONE(ast_, AST_LOCAL, Local);
  ------------------
  |  |  355|   314k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 314k, False: 13.2M]
  |  |  ------------------
  |  |  356|   314k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   314k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   314k|     var = alloc.clone(ast); \
  |  |  359|   314k|   } break
  ------------------
  |  Branch (386:9): [True: 314k, False: 0]
  ------------------
  387|      0|        CLONE(ast_, AST_OBJECT, Object);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 13.5M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (387:9): [True: 0, False: 0]
  ------------------
  388|      0|        CLONE(ast_, AST_OBJECT_COMPREHENSION, ObjectComprehension);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 13.5M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (388:9): [True: 0, False: 0]
  ------------------
  389|  44.8k|        CLONE(ast_, AST_OBJECT_COMPREHENSION_SIMPLE, ObjectComprehensionSimple);
  ------------------
  |  |  355|  44.8k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 44.8k, False: 13.5M]
  |  |  ------------------
  |  |  356|  44.8k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  44.8k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  44.8k|     var = alloc.clone(ast); \
  |  |  359|  44.8k|   } break
  ------------------
  |  Branch (389:9): [True: 44.8k, False: 0]
  ------------------
  390|      0|        CLONE(ast_, AST_PARENS, Parens);
  ------------------
  |  |  355|      0|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 0, False: 13.5M]
  |  |  ------------------
  |  |  356|      0|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|      0|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|      0|     var = alloc.clone(ast); \
  |  |  359|      0|   } break
  ------------------
  |  Branch (390:9): [True: 0, False: 0]
  ------------------
  391|  15.3k|        CLONE(ast_, AST_SELF, Self);
  ------------------
  |  |  355|  15.3k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 15.3k, False: 13.5M]
  |  |  ------------------
  |  |  356|  15.3k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  15.3k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  15.3k|     var = alloc.clone(ast); \
  |  |  359|  15.3k|   } break
  ------------------
  |  Branch (391:9): [True: 15.3k, False: 0]
  ------------------
  392|   148k|        CLONE(ast_, AST_SUPER_INDEX, SuperIndex);
  ------------------
  |  |  355|   148k|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 148k, False: 13.4M]
  |  |  ------------------
  |  |  356|   148k|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|   148k|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|   148k|     var = alloc.clone(ast); \
  |  |  359|   148k|   } break
  ------------------
  |  Branch (392:9): [True: 148k, False: 0]
  ------------------
  393|  2.86M|        CLONE(ast_, AST_UNARY, Unary);
  ------------------
  |  |  355|  2.86M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 2.86M, False: 10.7M]
  |  |  ------------------
  |  |  356|  2.86M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  2.86M|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  2.86M|     var = alloc.clone(ast); \
  |  |  359|  2.86M|   } break
  ------------------
  |  Branch (393:9): [True: 2.86M, False: 0]
  ------------------
  394|  3.27M|        CLONE(ast_, AST_VAR, Var);
  ------------------
  |  |  355|  3.27M|   case astType: { \
  |  |  ------------------
  |  |  |  Branch (355:4): [True: 3.27M, False: 10.3M]
  |  |  ------------------
  |  |  356|  3.27M|     assert(dynamic_cast<astClass *>(var)); \
  |  |  357|  3.27M|     auto *ast = static_cast<astClass *>(var); \
  |  |  358|  3.27M|     var = alloc.clone(ast); \
  |  |  359|  3.27M|   } break
  ------------------
  |  Branch (394:9): [True: 3.27M, False: 0]
  ------------------
  395|      0|        default:
  ------------------
  |  Branch (395:9): [True: 0, False: 13.5M]
  ------------------
  396|      0|            std::cerr << "INTERNAL ERROR: Unknown AST: " << ast_ << std::endl;
  397|      0|            std::abort();
  398|      0|            break;
  399|  13.5M|    }
  400|       |
  401|  13.5M|    CompilerPass::expr(ast_);
  402|  13.5M|}
_ZN7jsonnet8internal9clone_astERNS0_9AllocatorEPNS0_3ASTE:
  405|  29.7k|{
  406|  29.7k|    AST *r = ast;
  407|  29.7k|    ClonePass(alloc).expr(r);
  408|  29.7k|    return r;
  409|  29.7k|}
_ZN7jsonnet8internal9ClonePassC2ERNS0_9AllocatorE:
  350|  29.7k|    ClonePass(Allocator &alloc) : CompilerPass(alloc) {}

_ZN7jsonnet8internal12CompilerPassC2ERNS0_9AllocatorE:
   32|  39.6k|    CompilerPass(Allocator &alloc) : alloc(alloc) {}
_ZN7jsonnet8internal12CompilerPass13fodderElementERNS0_13FodderElementE:
   34|  83.6k|    virtual void fodderElement(FodderElement &) {}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_14LiteralBooleanE:
   84|  93.3k|    virtual void visit(LiteralBoolean *) {}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_13LiteralNumberE:
   86|  1.12M|    virtual void visit(LiteralNumber *) {}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_11LiteralNullE:
   90|   107k|    virtual void visit(LiteralNull *) {}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_4SelfE:
  102|  15.3k|    virtual void visit(Self *) {}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_3VarE:
  108|  3.60M|    virtual void visit(Var *) {}
_ZN7jsonnet8internal12CompilerPass5visitEPNS0_13LiteralStringE:
   88|  1.74M|    virtual void visit(LiteralString *) {}

_ZN7jsonnet8internal32path_dir_with_trailing_separatorERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
   31|     34|std::string path_dir_with_trailing_separator(const std::string &path) {
   32|     34|    size_t last_slash = path.find_last_of(DIR_SEPARATORS);
   33|     34|    if (last_slash != std::string::npos) {
  ------------------
  |  Branch (33:9): [True: 0, False: 34]
  ------------------
   34|      0|        return path.substr(0, last_slash + 1);
   35|      0|    }
   36|     34|    return "";
   37|     34|}

vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14HeapC2Ejd:
  346|  3.46k|        : gcTuneMinObjects(gc_tune_min_objects),
  347|  3.46k|          gcTuneGrowthTrigger(gc_tune_growth_trigger),
  348|  3.46k|          lastMark(0),
  349|  3.46k|          lastNumEntities(0),
  350|  3.46k|          numEntities(0)
  351|  3.46k|    {
  352|  3.46k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_18type_strENS1_5Value4TypeE:
   76|  1.54k|{
   77|  1.54k|    switch (t) {
   78|     15|        case Value::NULL_TYPE: return "null";
  ------------------
  |  Branch (78:9): [True: 15, False: 1.53k]
  ------------------
   79|    326|        case Value::BOOLEAN: return "boolean";
  ------------------
  |  Branch (79:9): [True: 326, False: 1.22k]
  ------------------
   80|    272|        case Value::NUMBER: return "number";
  ------------------
  |  Branch (80:9): [True: 272, False: 1.27k]
  ------------------
   81|     39|        case Value::ARRAY: return "array";
  ------------------
  |  Branch (81:9): [True: 39, False: 1.50k]
  ------------------
   82|      1|        case Value::FUNCTION: return "function";
  ------------------
  |  Branch (82:9): [True: 1, False: 1.54k]
  ------------------
   83|    263|        case Value::OBJECT: return "object";
  ------------------
  |  Branch (83:9): [True: 263, False: 1.28k]
  ------------------
   84|    632|        case Value::STRING: return "string";
  ------------------
  |  Branch (84:9): [True: 632, False: 916]
  ------------------
   85|      0|        default:
  ------------------
  |  Branch (85:9): [True: 0, False: 1.54k]
  ------------------
   86|      0|            std::cerr << "INTERNAL ERROR: Unknown type: " << t << std::endl;
   87|      0|            std::abort();
   88|      0|            return "";  // Quiet, compiler.
   89|  1.54k|    }
   90|  1.54k|}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_18type_strERKNS1_5ValueE:
   94|    417|{
   95|    417|    return type_str(v.t);
   96|    417|}
vm.cpp:_ZNK7jsonnet8internal12_GLOBAL__N_15Value6isHeapEv:
   69|  46.4M|    {
   70|  46.4M|        return t & 0x10;
   71|  46.4M|    }
vm.cpp:_ZNK7jsonnet8internal12_GLOBAL__N_111HeapClosure9isBuiltinEv:
  293|  22.5M|    bool isBuiltin() const { return !this->body || this->body->type == AST_BUILTIN_FUNCTION_BODY; }
  ------------------
  |  Branch (293:37): [True: 0, False: 22.5M]
  |  Branch (293:52): [True: 16.7M, False: 5.79M]
  ------------------
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_19HeapThunkC2EPKNS0_10IdentifierEPNS1_10HeapObjectEjPKNS0_3ASTE:
  140|  32.0M|        : HeapEntity(THUNK), filled(false), name(name), self(self), offset(offset), body(body)
  141|  32.0M|    {
  142|  32.0M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_110HeapEntityC2ENS2_4TypeE:
   43|  71.1M|    HeapEntity(Type type_) : type(type_) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_110HeapEntityD2Ev:
   44|  71.1M|    virtual ~HeapEntity() {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap9checkHeapEv:
  489|  71.1M|    {
  490|  71.1M|        return numEntities > gcTuneMinObjects &&
  ------------------
  |  Branch (490:16): [True: 29.9M, False: 41.2M]
  ------------------
  491|  29.9M|               numEntities > gcTuneGrowthTrigger * lastNumEntities;
  ------------------
  |  Branch (491:16): [True: 78.7k, False: 29.8M]
  ------------------
  492|  71.1M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap8markFromEPNS1_10HeapEntityE:
  369|  66.0M|    {
  370|  66.0M|        assert(from != nullptr);
  ------------------
  |  Branch (370:9): [True: 66.0M, False: 0]
  ------------------
  371|  66.0M|        const GarbageCollectionMark thisMark = lastMark + 1;
  372|  66.0M|        struct State {
  373|  66.0M|            HeapEntity *ent;
  374|  66.0M|            std::vector<HeapEntity *> children;
  375|  66.0M|            State(HeapEntity *ent) : ent(ent) {}
  376|  66.0M|        };
  377|       |
  378|  66.0M|        std::vector<State> stack;
  379|  66.0M|        stack.emplace_back(from);
  380|       |
  381|   269M|        while (stack.size() > 0) {
  ------------------
  |  Branch (381:16): [True: 203M, False: 66.0M]
  ------------------
  382|   203M|            size_t curr_index = stack.size() - 1;
  383|   203M|            State &s = stack[curr_index];
  384|   203M|            HeapEntity *curr = s.ent;
  385|   203M|            if (curr->mark != thisMark) {
  ------------------
  |  Branch (385:17): [True: 50.7M, False: 153M]
  ------------------
  386|  50.7M|                curr->mark = thisMark;
  387|       |
  388|  50.7M|                switch(curr->type) {
  389|  7.80M|                    case HeapEntity::SIMPLE_OBJECT: {
  ------------------
  |  Branch (389:21): [True: 7.80M, False: 42.9M]
  ------------------
  390|  7.80M|                        assert(dynamic_cast<HeapSimpleObject *>(curr));
  ------------------
  |  Branch (390:25): [True: 7.80M, False: 0]
  ------------------
  391|  7.80M|                        auto *obj = static_cast<HeapSimpleObject *>(curr);
  392|  7.80M|                        for (auto upv : obj->upValues)
  ------------------
  |  Branch (392:39): [True: 8.41M, False: 7.80M]
  ------------------
  393|  8.41M|                            addIfHeapEntity(upv.second, s.children);
  394|  7.80M|                        break;
  395|  7.80M|                    }
  396|  3.87M|                    case HeapEntity::EXTENDED_OBJECT: {
  ------------------
  |  Branch (396:21): [True: 3.87M, False: 46.9M]
  ------------------
  397|  3.87M|                        assert(dynamic_cast<HeapExtendedObject *>(curr));
  ------------------
  |  Branch (397:25): [True: 3.87M, False: 0]
  ------------------
  398|  3.87M|                        auto *obj = static_cast<HeapExtendedObject *>(curr);
  399|  3.87M|                        addIfHeapEntity(obj->left, s.children);
  400|  3.87M|                        addIfHeapEntity(obj->right, s.children);
  401|  3.87M|                        break;
  402|  3.87M|                    }
  403|      0|                    case HeapEntity::RESTRICTED_OBJECT: {
  ------------------
  |  Branch (403:21): [True: 0, False: 50.7M]
  ------------------
  404|      0|                        assert(dynamic_cast<HeapRestrictedObject *>(curr));
  ------------------
  |  Branch (404:25): [True: 0, False: 0]
  ------------------
  405|      0|                        auto *obj = static_cast<HeapRestrictedObject *>(curr);
  406|      0|                        addIfHeapEntity(obj->obj, s.children);
  407|      0|                        break;
  408|      0|                    }
  409|      0|                    case HeapEntity::COMPREHENSION_OBJECT: {
  ------------------
  |  Branch (409:21): [True: 0, False: 50.7M]
  ------------------
  410|      0|                        assert(dynamic_cast<HeapComprehensionObject *>(curr));
  ------------------
  |  Branch (410:25): [True: 0, False: 0]
  ------------------
  411|      0|                        auto *obj = static_cast<HeapComprehensionObject *>(curr);
  412|      0|                        for (auto upv : obj->upValues)
  ------------------
  |  Branch (412:39): [True: 0, False: 0]
  ------------------
  413|      0|                            addIfHeapEntity(upv.second, s.children);
  414|      0|                        for (auto upv : obj->compValues)
  ------------------
  |  Branch (414:39): [True: 0, False: 0]
  ------------------
  415|      0|                            addIfHeapEntity(upv.second, s.children);
  416|      0|                        break;
  417|      0|                    }
  418|  1.33M|                    case HeapEntity::ARRAY: {
  ------------------
  |  Branch (418:21): [True: 1.33M, False: 49.4M]
  ------------------
  419|  1.33M|                        assert(dynamic_cast<HeapArray *>(curr));
  ------------------
  |  Branch (419:25): [True: 1.33M, False: 0]
  ------------------
  420|  1.33M|                        auto *arr = static_cast<HeapArray *>(curr);
  421|  1.33M|                        for (auto el : arr->elements)
  ------------------
  |  Branch (421:38): [True: 12.6M, False: 1.33M]
  ------------------
  422|  12.6M|                            addIfHeapEntity(el, s.children);
  423|  1.33M|                        break;
  424|  1.33M|                    }
  425|  1.21M|                    case HeapEntity::CLOSURE: {
  ------------------
  |  Branch (425:21): [True: 1.21M, False: 49.5M]
  ------------------
  426|  1.21M|                        assert(dynamic_cast<HeapClosure *>(curr));
  ------------------
  |  Branch (426:25): [True: 1.21M, False: 0]
  ------------------
  427|  1.21M|                        auto *func = static_cast<HeapClosure *>(curr);
  428|  1.21M|                        for (auto upv : func->upValues)
  ------------------
  |  Branch (428:39): [True: 2.48M, False: 1.21M]
  ------------------
  429|  2.48M|                            addIfHeapEntity(upv.second, s.children);
  430|  1.21M|                        if (func->self)
  ------------------
  |  Branch (430:29): [True: 931k, False: 279k]
  ------------------
  431|   931k|                            addIfHeapEntity(func->self, s.children);
  432|  1.21M|                        break;
  433|  1.21M|                    }
  434|  35.1M|                    case HeapEntity::THUNK: {
  ------------------
  |  Branch (434:21): [True: 35.1M, False: 15.6M]
  ------------------
  435|  35.1M|                        assert(dynamic_cast<HeapThunk *>(curr));
  ------------------
  |  Branch (435:25): [True: 35.1M, False: 0]
  ------------------
  436|  35.1M|                        auto *thunk = static_cast<HeapThunk *>(curr);
  437|  35.1M|                        if (thunk->filled) {
  ------------------
  |  Branch (437:29): [True: 9.48M, False: 25.7M]
  ------------------
  438|  9.48M|                            if (thunk->content.isHeap())
  ------------------
  |  Branch (438:33): [True: 5.62M, False: 3.85M]
  ------------------
  439|  5.62M|                                addIfHeapEntity(thunk->content.v.h, s.children);
  440|  25.7M|                        } else {
  441|  25.7M|                            for (auto upv : thunk->upValues)
  ------------------
  |  Branch (441:43): [True: 6.55M, False: 25.7M]
  ------------------
  442|  6.55M|                                addIfHeapEntity(upv.second, s.children);
  443|  25.7M|                            if (thunk->self)
  ------------------
  |  Branch (443:33): [True: 24.5M, False: 1.16M]
  ------------------
  444|  24.5M|                                addIfHeapEntity(thunk->self, s.children);
  445|  25.7M|                        }
  446|  35.1M|                        break;
  447|  35.1M|                    }
  448|  1.38M|                    case HeapEntity::STRING:
  ------------------
  |  Branch (448:21): [True: 1.38M, False: 49.4M]
  ------------------
  449|  1.38M|                        assert(dynamic_cast<HeapString *>(curr));
  ------------------
  |  Branch (449:25): [True: 1.38M, False: 0]
  ------------------
  450|  1.38M|                        break;
  451|  1.38M|                    default:
  ------------------
  |  Branch (451:21): [True: 0, False: 50.7M]
  ------------------
  452|      0|                        assert(false);
  ------------------
  |  Branch (452:25): [Folded, False: 0]
  ------------------
  453|      0|                        break;
  454|  50.7M|                }
  455|  50.7M|            }
  456|       |
  457|   203M|            if (s.children.size() > 0) {
  ------------------
  |  Branch (457:17): [True: 68.9M, False: 134M]
  ------------------
  458|  68.9M|                HeapEntity *next = s.children[s.children.size() - 1];
  459|  68.9M|                s.children.pop_back();
  460|  68.9M|                stack.emplace_back(next);  // CAUTION: s invalidated here
  461|   134M|            } else {
  462|   134M|                stack.pop_back();  // CAUTION: s invalidated here
  463|   134M|            }
  464|   203M|        }
  465|  66.0M|    }
vm.cpp:_ZZN7jsonnet8internal12_GLOBAL__N_14Heap8markFromEPNS1_10HeapEntityEEN5StateC2ES4_:
  375|   134M|            State(HeapEntity *ent) : ent(ent) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap15addIfHeapEntityEPNS1_10HeapEntityERNSt3__16vectorIS4_NS5_9allocatorIS4_EEEE:
  340|  68.9M|    {
  341|  68.9M|        vec.push_back(v);
  342|  68.9M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap8markFromENS1_5ValueE:
  362|  36.6M|    {
  363|  36.6M|        if (v.isHeap())
  ------------------
  |  Branch (363:13): [True: 9.20M, False: 27.3M]
  ------------------
  364|  9.20M|            markFrom(v.v.h);
  365|  36.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap5sweepEv:
  469|  82.2k|    {
  470|  82.2k|        lastMark++;
  471|       |        // Heap shrinks during this loop.  Do not cache entities.size().
  472|   122M|        for (unsigned long i = 0; i < entities.size(); ++i) {
  ------------------
  |  Branch (472:35): [True: 121M, False: 82.2k]
  ------------------
  473|   121M|            HeapEntity *x = entities[i];
  474|   121M|            if (x->mark != lastMark) {
  ------------------
  |  Branch (474:17): [True: 71.1M, False: 50.7M]
  ------------------
  475|  71.1M|                delete x;
  476|  71.1M|                if (i != entities.size() - 1) {
  ------------------
  |  Branch (476:21): [True: 71.1M, False: 69.3k]
  ------------------
  477|       |                    // Swap it with the back.
  478|  71.1M|                    entities[i] = entities[entities.size() - 1];
  479|  71.1M|                }
  480|  71.1M|                entities.pop_back();
  481|  71.1M|                --i;
  482|  71.1M|            }
  483|   121M|        }
  484|  82.2k|        lastNumEntities = numEntities = entities.size();
  485|  82.2k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_19HeapThunk4fillERKNS1_5ValueE:
  145|  18.4M|    {
  146|  18.4M|        content = v;
  147|  18.4M|        filled = true;
  148|  18.4M|        self = nullptr;
  149|  18.4M|        upValues.clear();
  150|  18.4M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapArrayEJRKNSt3__16vectorIPNS1_9HeapThunkENS5_9allocatorIS8_EEEEEEEPT_DpOT0_:
  501|  1.12M|    {
  502|  1.12M|        T *r = new T(std::forward<Args>(args)...);
  503|  1.12M|        entities.push_back(r);
  504|  1.12M|        r->mark = lastMark;
  505|  1.12M|        numEntities = entities.size();
  506|  1.12M|        return r;
  507|  1.12M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_19HeapArrayC2ERKNSt3__16vectorIPNS1_9HeapThunkENS3_9allocatorIS6_EEEE:
  159|  1.12M|        : HeapEntity(ARRAY), elements(elements)
  160|  1.12M|    {
  161|  1.12M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_10HeapStringEJRKNSt3__112basic_stringIDiNS5_11char_traitsIDiEENS5_9allocatorIDiEEEEEEEPT_DpOT0_:
  501|  27.6M|    {
  502|  27.6M|        T *r = new T(std::forward<Args>(args)...);
  503|  27.6M|        entities.push_back(r);
  504|  27.6M|        r->mark = lastMark;
  505|  27.6M|        numEntities = entities.size();
  506|  27.6M|        return r;
  507|  27.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_110HeapStringC2ERKNSt3__112basic_stringIDiNS3_11char_traitsIDiEENS3_9allocatorIDiEEEE:
  299|  27.6M|    HeapString(const UString &value) : HeapEntity(STRING), value(value) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJRPKNS0_10IdentifierEDniDnEEEPT_DpOT0_:
  501|   272k|    {
  502|   272k|        T *r = new T(std::forward<Args>(args)...);
  503|   272k|        entities.push_back(r);
  504|   272k|        r->mark = lastMark;
  505|   272k|        numEntities = entities.size();
  506|   272k|        return r;
  507|   272k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_114HeapLeafObjectC2ENS1_10HeapEntity4TypeE:
  166|  1.43M|    HeapLeafObject(Type type) : HeapObject(type) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_110HeapObjectC2ENS1_10HeapEntity4TypeE:
  109|  2.31M|    HeapObject(Type type) : HeapEntity(type) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111HeapClosure5ParamC2EPKNS0_10IdentifierEPKNS0_3ASTE:
  275|  13.2M|        Param(const Identifier *id, const AST *def) : id(id), def(def) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_11HeapClosureEJNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEEDniRNS5_6vectorINS4_5ParamENSE_ISL_EEEERPKNS0_19BuiltinFunctionBodyERKNS5_12basic_stringIcNS5_11char_traitsIcEENSE_IcEEEEEEEPT_DpOT0_:
  501|  6.49M|    {
  502|  6.49M|        T *r = new T(std::forward<Args>(args)...);
  503|  6.49M|        entities.push_back(r);
  504|  6.49M|        r->mark = lastMark;
  505|  6.49M|        numEntities = entities.size();
  506|  6.49M|        return r;
  507|  6.49M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111HeapClosureC2ERKNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS3_4lessIS7_EENS3_9allocatorINS3_4pairIKS7_S9_EEEEEEPNS1_10HeapObjectEjRKNS3_6vectorINS2_5ParamENSC_ISN_EEEEPKNS0_3ASTERKNS3_12basic_stringIcNS3_11char_traitsIcEENSC_IcEEEE:
  283|  8.10M|        : HeapEntity(CLOSURE),
  284|  8.10M|          upValues(up_values),
  285|  8.10M|          self(self),
  286|  8.10M|          offset(offset),
  287|  8.10M|          params(params),
  288|  8.10M|          body(body),
  289|  8.10M|          builtinName(builtin_name)
  290|  8.10M|    {
  291|  8.10M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJDnDniPKNS0_3ASTEEEEPT_DpOT0_:
  501|  3.46k|    {
  502|  3.46k|        T *r = new T(std::forward<Args>(args)...);
  503|  3.46k|        entities.push_back(r);
  504|  3.46k|        r->mark = lastMark;
  505|  3.46k|        numEntities = entities.size();
  506|  3.46k|        return r;
  507|  3.46k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJPNS0_10IdentifierERPNS1_10HeapObjectEiRKPNS0_3ASTEEEEPT_DpOT0_:
  501|   557k|    {
  502|   557k|        T *r = new T(std::forward<Args>(args)...);
  503|   557k|        entities.push_back(r);
  504|   557k|        r->mark = lastMark;
  505|   557k|        numEntities = entities.size();
  506|   557k|        return r;
  507|   557k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14HeapD2Ev:
  355|  3.46k|    {
  356|       |        // Nothing is marked, everything will be collected.
  357|  3.46k|        sweep();
  358|  3.46k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJRPKNS0_10IdentifierERPNS1_10HeapObjectERjRKPNS0_3ASTEEEEPT_DpOT0_:
  501|  16.4M|    {
  502|  16.4M|        T *r = new T(std::forward<Args>(args)...);
  503|  16.4M|        entities.push_back(r);
  504|  16.4M|        r->mark = lastMark;
  505|  16.4M|        numEntities = entities.size();
  506|  16.4M|        return r;
  507|  16.4M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_11HeapClosureEJRKNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEERPNS1_10HeapObjectERjRKNS5_6vectorINS4_5ParamENSE_ISR_EEEERPNS0_3ASTERA1_KcEEEPT_DpOT0_:
  501|  1.60M|    {
  502|  1.60M|        T *r = new T(std::forward<Args>(args)...);
  503|  1.60M|        entities.push_back(r);
  504|  1.60M|        r->mark = lastMark;
  505|  1.60M|        numEntities = entities.size();
  506|  1.60M|        return r;
  507|  1.60M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJRKPKNS0_10IdentifierERPNS1_10HeapObjectERjRKPNS0_3ASTEEEEPT_DpOT0_:
  501|  14.6M|    {
  502|  14.6M|        T *r = new T(std::forward<Args>(args)...);
  503|  14.6M|        entities.push_back(r);
  504|  14.6M|        r->mark = lastMark;
  505|  14.6M|        numEntities = entities.size();
  506|  14.6M|        return r;
  507|  14.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_16HeapSimpleObjectEJRNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEERNS6_IS9_NS4_5FieldESD_NSE_INSF_ISG_SL_EEEEEERNS5_4listIPNS0_3ASTENSE_ISS_EEEEEEEPT_DpOT0_:
  501|  1.43M|    {
  502|  1.43M|        T *r = new T(std::forward<Args>(args)...);
  503|  1.43M|        entities.push_back(r);
  504|  1.43M|        r->mark = lastMark;
  505|  1.43M|        numEntities = entities.size();
  506|  1.43M|        return r;
  507|  1.43M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_116HeapSimpleObjectC2ERKNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS3_4lessIS7_EENS3_9allocatorINS3_4pairIKS7_S9_EEEEEENS4_IS7_NS2_5FieldESB_NSC_INSD_ISE_SK_EEEEEENS3_4listIPNS0_3ASTENSC_ISQ_EEEE:
  196|  1.43M|        : HeapLeafObject(SIMPLE_OBJECT), upValues(up_values), fields(fields), asserts(asserts)
  197|  1.43M|    {
  198|  1.43M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJRPKNS0_10IdentifierERPNS1_10HeapObjectERjRKPKNS0_3ASTEEEEPT_DpOT0_:
  501|     29|    {
  502|     29|        T *r = new T(std::forward<Args>(args)...);
  503|     29|        entities.push_back(r);
  504|     29|        r->mark = lastMark;
  505|     29|        numEntities = entities.size();
  506|     29|        return r;
  507|     29|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_18HeapExtendedObjectEJRPNS1_10HeapObjectES7_EEEPT_DpOT0_:
  501|   881k|    {
  502|   881k|        T *r = new T(std::forward<Args>(args)...);
  503|   881k|        entities.push_back(r);
  504|   881k|        r->mark = lastMark;
  505|   881k|        numEntities = entities.size();
  506|   881k|        return r;
  507|   881k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_118HeapExtendedObjectC2EPNS1_10HeapObjectES4_:
  210|   881k|        : HeapObject(EXTENDED_OBJECT), left(left), right(right)
  211|   881k|    {
  212|   881k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_14Heap10makeEntityINS1_9HeapThunkEJRPKNS0_10IdentifierERPNS1_10HeapObjectERjRPNS0_3ASTEEEEPT_DpOT0_:
  501|  81.0k|    {
  502|  81.0k|        T *r = new T(std::forward<Args>(args)...);
  503|  81.0k|        entities.push_back(r);
  504|  81.0k|        r->mark = lastMark;
  505|  81.0k|        numEntities = entities.size();
  506|  81.0k|        return r;
  507|  81.0k|    }

_ZN7jsonnet8internal23jsonnet_static_analysisEPNS0_3ASTE:
  228|  7.56k|{
  229|  7.56k|    static_analysis(ast, false, IdSet{});
  230|  7.56k|}
static_analysis.cpp:_ZN7jsonnet8internalL15static_analysisEPNS0_3ASTEbRKNSt3__13setIPKNS0_10IdentifierENS3_4lessIS7_EENS3_9allocatorIS7_EEEE:
   41|   238M|{
   42|   238M|    IdSet r;
   43|       |
   44|   238M|    switch (ast_->type) {
   45|  21.4M|    case AST_APPLY: {
  ------------------
  |  Branch (45:5): [True: 21.4M, False: 216M]
  ------------------
   46|  21.4M|        assert(dynamic_cast<Apply *>(ast_));
  ------------------
  |  Branch (46:9): [True: 21.4M, False: 0]
  ------------------
   47|  21.4M|        auto* ast = static_cast<Apply *>(ast_);
   48|  21.4M|        append(r, static_analysis(ast->target, in_object, vars));
   49|  21.4M|        for (const auto &arg : ast->args)
  ------------------
  |  Branch (49:30): [True: 35.2M, False: 21.4M]
  ------------------
   50|  35.2M|            append(r, static_analysis(arg.expr, in_object, vars));
   51|  21.4M|    } break;
   52|      0|    case AST_APPLY_BRACE: {
  ------------------
  |  Branch (52:5): [True: 0, False: 238M]
  ------------------
   53|      0|        assert(dynamic_cast<ApplyBrace *>(ast_));
  ------------------
  |  Branch (53:9): [True: 0, False: 0]
  ------------------
   54|       |        // Nothing to do.
   55|      0|    } break;
   56|  5.21M|    case AST_ARRAY: {
  ------------------
  |  Branch (56:5): [True: 5.21M, False: 233M]
  ------------------
   57|  5.21M|        assert(dynamic_cast<Array *>(ast_));
  ------------------
  |  Branch (57:9): [True: 5.21M, False: 0]
  ------------------
   58|  5.21M|        auto* ast = static_cast<Array *>(ast_);
   59|  5.21M|        for (auto &el : ast->elements)
  ------------------
  |  Branch (59:23): [True: 6.59M, False: 5.21M]
  ------------------
   60|  6.59M|            append(r, static_analysis(el.expr, in_object, vars));
   61|  5.21M|    } break;
   62|  21.8M|    case AST_BINARY: {
  ------------------
  |  Branch (62:5): [True: 21.8M, False: 216M]
  ------------------
   63|  21.8M|        assert(dynamic_cast<Binary *>(ast_));
  ------------------
  |  Branch (63:9): [True: 21.8M, False: 0]
  ------------------
   64|  21.8M|        auto* ast = static_cast<Binary *>(ast_);
   65|  21.8M|        append(r, static_analysis(ast->left, in_object, vars));
   66|  21.8M|        append(r, static_analysis(ast->right, in_object, vars));
   67|  21.8M|    } break;
   68|   317k|    case AST_BUILTIN_FUNCTION: {
  ------------------
  |  Branch (68:5): [True: 317k, False: 237M]
  ------------------
   69|   317k|        assert(dynamic_cast<BuiltinFunction *>(ast_));
  ------------------
  |  Branch (69:9): [True: 317k, False: 0]
  ------------------
   70|       |        // Nothing to do.
   71|   317k|    } break;
   72|  8.80M|    case AST_CONDITIONAL: {
  ------------------
  |  Branch (72:5): [True: 8.80M, False: 229M]
  ------------------
   73|  8.80M|        assert(dynamic_cast<Conditional *>(ast_));
  ------------------
  |  Branch (73:9): [True: 8.80M, False: 0]
  ------------------
   74|  8.80M|        auto* ast = static_cast<Conditional *>(ast_);
   75|  8.80M|        append(r, static_analysis(ast->cond, in_object, vars));
   76|  8.80M|        append(r, static_analysis(ast->branchTrue, in_object, vars));
   77|  8.80M|        append(r, static_analysis(ast->branchFalse, in_object, vars));
   78|  8.80M|    } break;
   79|  4.64M|    case AST_ERROR: {
  ------------------
  |  Branch (79:5): [True: 4.64M, False: 233M]
  ------------------
   80|  4.64M|        assert(dynamic_cast<Error *>(ast_));
  ------------------
  |  Branch (80:9): [True: 4.64M, False: 0]
  ------------------
   81|  4.64M|        auto* ast = static_cast<Error *>(ast_);
   82|  4.64M|        append(r, static_analysis(ast->expr, in_object, vars));
   83|  4.64M|    } break;
   84|  5.40M|    case AST_FUNCTION: {
  ------------------
  |  Branch (84:5): [True: 5.40M, False: 232M]
  ------------------
   85|  5.40M|        assert(dynamic_cast<Function *>(ast_));
  ------------------
  |  Branch (85:9): [True: 5.40M, False: 0]
  ------------------
   86|  5.40M|        auto* ast = static_cast<Function *>(ast_);
   87|  5.40M|        auto new_vars = vars;
   88|  5.40M|        IdSet params;
   89|  9.84M|        for (const auto &p : ast->params) {
  ------------------
  |  Branch (89:28): [True: 9.84M, False: 5.40M]
  ------------------
   90|  9.84M|            if (params.find(p.id) != params.end()) {
  ------------------
  |  Branch (90:17): [True: 5, False: 9.84M]
  ------------------
   91|      5|                std::string msg = "Duplicate function parameter: " + encode_utf8(p.id->name);
   92|      5|                throw StaticError(ast_->location, msg);
   93|      5|            }
   94|  9.84M|            params.insert(p.id);
   95|  9.84M|            new_vars.insert(p.id);
   96|  9.84M|        }
   97|       |
   98|  5.40M|        auto fv = static_analysis(ast->body, in_object, new_vars);
   99|  9.83M|        for (const auto &p : ast->params) {
  ------------------
  |  Branch (99:28): [True: 9.83M, False: 5.40M]
  ------------------
  100|  9.83M|            if (p.expr != nullptr)
  ------------------
  |  Branch (100:17): [True: 158k, False: 9.68M]
  ------------------
  101|   158k|                append(fv, static_analysis(p.expr, in_object, new_vars));
  102|  9.83M|        }
  103|  5.40M|        for (const auto &p : ast->params)
  ------------------
  |  Branch (103:28): [True: 9.83M, False: 5.40M]
  ------------------
  104|  9.83M|            fv.erase(p.id);
  105|  5.40M|        append(r, fv);
  106|  5.40M|    } break;
  107|    320|    case AST_IMPORT: {
  ------------------
  |  Branch (107:5): [True: 320, False: 238M]
  ------------------
  108|    320|        assert(dynamic_cast<Import *>(ast_));
  ------------------
  |  Branch (108:9): [True: 320, False: 0]
  ------------------
  109|       |        // Nothing to do.
  110|    320|    } break;
  111|    320|    case AST_IMPORTSTR: {
  ------------------
  |  Branch (111:5): [True: 77, False: 238M]
  ------------------
  112|     77|        assert(dynamic_cast<Importstr *>(ast_));
  ------------------
  |  Branch (112:9): [True: 77, False: 0]
  ------------------
  113|       |        // Nothing to do.
  114|     77|    } break;
  115|  20.6k|    case AST_IMPORTBIN: {
  ------------------
  |  Branch (115:5): [True: 20.6k, False: 238M]
  ------------------
  116|  20.6k|        assert(dynamic_cast<Importbin *>(ast_));
  ------------------
  |  Branch (116:9): [True: 20.6k, False: 0]
  ------------------
  117|       |        // Nothing to do.
  118|  20.6k|    } break;
  119|  50.4k|    case AST_IN_SUPER: {
  ------------------
  |  Branch (119:5): [True: 50.4k, False: 238M]
  ------------------
  120|  50.4k|        assert(dynamic_cast<const InSuper *>(ast_));
  ------------------
  |  Branch (120:9): [True: 50.4k, False: 0]
  ------------------
  121|  50.4k|        auto* ast = static_cast<const InSuper *>(ast_);
  122|  50.4k|        if (!in_object)
  ------------------
  |  Branch (122:13): [True: 21, False: 50.4k]
  ------------------
  123|     21|            throw StaticError(ast_->location, "Can't use super outside of an object.");
  124|  50.4k|        append(r, static_analysis(ast->element, in_object, vars));
  125|  50.4k|    } break;
  126|  23.4M|    case AST_INDEX: {
  ------------------
  |  Branch (126:5): [True: 23.4M, False: 214M]
  ------------------
  127|  23.4M|        assert(dynamic_cast<const Index *>(ast_));
  ------------------
  |  Branch (127:9): [True: 23.4M, False: 0]
  ------------------
  128|  23.4M|        auto* ast = static_cast<const Index *>(ast_);
  129|  23.4M|        append(r, static_analysis(ast->target, in_object, vars));
  130|  23.4M|        append(r, static_analysis(ast->index, in_object, vars));
  131|  23.4M|    } break;
  132|  11.3M|    case AST_LOCAL: {
  ------------------
  |  Branch (132:5): [True: 11.3M, False: 226M]
  ------------------
  133|  11.3M|        assert(dynamic_cast<const Local *>(ast_));
  ------------------
  |  Branch (133:9): [True: 11.3M, False: 0]
  ------------------
  134|  11.3M|        auto* ast = static_cast<const Local *>(ast_);
  135|  11.3M|        IdSet ast_vars;
  136|  19.1M|        for (const auto &bind : ast->binds) {
  ------------------
  |  Branch (136:31): [True: 19.1M, False: 11.3M]
  ------------------
  137|  19.1M|            ast_vars.insert(bind.var);
  138|  19.1M|        }
  139|  11.3M|        auto new_vars = vars;
  140|  11.3M|        append(new_vars, ast_vars);
  141|  11.3M|        IdSet fvs;
  142|  19.0M|        for (const auto &bind : ast->binds) {
  ------------------
  |  Branch (142:31): [True: 19.0M, False: 11.3M]
  ------------------
  143|  19.0M|            append(fvs, static_analysis(bind.body, in_object, new_vars));
  144|  19.0M|        }
  145|       |
  146|  11.3M|        append(fvs, static_analysis(ast->body, in_object, new_vars));
  147|       |
  148|  11.3M|        for (const auto &bind : ast->binds)
  ------------------
  |  Branch (148:31): [True: 19.0M, False: 11.3M]
  ------------------
  149|  19.0M|            fvs.erase(bind.var);
  150|       |
  151|  11.3M|        append(r, fvs);
  152|  11.3M|    } break;
  153|   715k|    case AST_LITERAL_BOOLEAN: {
  ------------------
  |  Branch (153:5): [True: 715k, False: 237M]
  ------------------
  154|   715k|        assert(dynamic_cast<const LiteralBoolean *>(ast_));
  ------------------
  |  Branch (154:9): [True: 715k, False: 0]
  ------------------
  155|       |        // Nothing to do.
  156|   715k|    } break;
  157|  15.2M|    case AST_LITERAL_NUMBER: {
  ------------------
  |  Branch (157:5): [True: 15.2M, False: 222M]
  ------------------
  158|  15.2M|        assert(dynamic_cast<const LiteralNumber *>(ast_));
  ------------------
  |  Branch (158:9): [True: 15.2M, False: 0]
  ------------------
  159|       |        // Nothing to do.
  160|  15.2M|    } break;
  161|  41.8M|    case AST_LITERAL_STRING: {
  ------------------
  |  Branch (161:5): [True: 41.8M, False: 196M]
  ------------------
  162|  41.8M|        assert(dynamic_cast<const LiteralString *>(ast_));
  ------------------
  |  Branch (162:9): [True: 41.8M, False: 0]
  ------------------
  163|       |        // Nothing to do.
  164|  41.8M|    } break;
  165|  41.8M|    case AST_LITERAL_NULL: {
  ------------------
  |  Branch (165:5): [True: 428k, False: 237M]
  ------------------
  166|   428k|        assert(dynamic_cast<const LiteralNull *>(ast_));
  ------------------
  |  Branch (166:9): [True: 428k, False: 0]
  ------------------
  167|       |        // Nothing to do.
  168|   428k|    } break;
  169|  1.45M|    case AST_DESUGARED_OBJECT: {
  ------------------
  |  Branch (169:5): [True: 1.45M, False: 236M]
  ------------------
  170|  1.45M|        assert(dynamic_cast<DesugaredObject *>(ast_));
  ------------------
  |  Branch (170:9): [True: 1.45M, False: 0]
  ------------------
  171|  1.45M|        auto* ast = static_cast<DesugaredObject *>(ast_);
  172|  6.76M|        for (auto &field : ast->fields) {
  ------------------
  |  Branch (172:26): [True: 6.76M, False: 1.45M]
  ------------------
  173|  6.76M|            append(r, static_analysis(field.name, in_object, vars));
  174|  6.76M|            append(r, static_analysis(field.body, true, vars));
  175|  6.76M|        }
  176|  1.45M|        for (AST *assert : ast->asserts) {
  ------------------
  |  Branch (176:26): [True: 102k, False: 1.45M]
  ------------------
  177|   102k|            append(r, static_analysis(assert, true, vars));
  178|   102k|        }
  179|  1.45M|    } break;
  180|   935k|    case AST_OBJECT_COMPREHENSION_SIMPLE: {
  ------------------
  |  Branch (180:5): [True: 935k, False: 237M]
  ------------------
  181|   935k|        assert(dynamic_cast<ObjectComprehensionSimple *>(ast_));
  ------------------
  |  Branch (181:9): [True: 935k, False: 0]
  ------------------
  182|   935k|        auto* ast = static_cast<ObjectComprehensionSimple *>(ast_);
  183|   935k|        auto new_vars = vars;
  184|   935k|        new_vars.insert(ast->id);
  185|   935k|        append(r, static_analysis(ast->field, false, new_vars));
  186|   935k|        append(r, static_analysis(ast->value, true, new_vars));
  187|   935k|        r.erase(ast->id);
  188|   935k|        append(r, static_analysis(ast->array, in_object, vars));
  189|   935k|    } break;
  190|  1.88M|    case AST_SELF: {
  ------------------
  |  Branch (190:5): [True: 1.88M, False: 236M]
  ------------------
  191|  1.88M|        assert(dynamic_cast<const Self *>(ast_));
  ------------------
  |  Branch (191:9): [True: 1.88M, False: 0]
  ------------------
  192|  1.88M|        if (!in_object)
  ------------------
  |  Branch (192:13): [True: 8, False: 1.88M]
  ------------------
  193|      8|            throw StaticError(ast_->location, "Can't use self outside of an object.");
  194|  1.88M|    } break;
  195|  1.88M|    case AST_SUPER_INDEX: {
  ------------------
  |  Branch (195:5): [True: 50.5k, False: 238M]
  ------------------
  196|  50.5k|        assert(dynamic_cast<const SuperIndex *>(ast_));
  ------------------
  |  Branch (196:9): [True: 50.5k, False: 0]
  ------------------
  197|  50.5k|        auto* ast = static_cast<const SuperIndex *>(ast_);
  198|  50.5k|        if (!in_object)
  ------------------
  |  Branch (198:13): [True: 8, False: 50.5k]
  ------------------
  199|      8|            throw StaticError(ast_->location, "Can't use super outside of an object.");
  200|  50.5k|        append(r, static_analysis(ast->index, in_object, vars));
  201|  50.5k|    } break;
  202|   755k|    case AST_UNARY: {
  ------------------
  |  Branch (202:5): [True: 755k, False: 237M]
  ------------------
  203|   755k|        assert(dynamic_cast<const Unary *>(ast_));
  ------------------
  |  Branch (203:9): [True: 755k, False: 0]
  ------------------
  204|   755k|        auto* ast = static_cast<const Unary *>(ast_);
  205|   755k|        append(r, static_analysis(ast->expr, in_object, vars));
  206|   755k|    } break;
  207|  72.3M|    case AST_VAR: {
  ------------------
  |  Branch (207:5): [True: 72.3M, False: 165M]
  ------------------
  208|  72.3M|        assert(dynamic_cast<const Var *>(ast_));
  ------------------
  |  Branch (208:9): [True: 72.3M, False: 0]
  ------------------
  209|  72.3M|        auto* ast = static_cast<const Var *>(ast_);
  210|  72.3M|        if (vars.find(ast->id) == vars.end()) {
  ------------------
  |  Branch (210:13): [True: 598, False: 72.3M]
  ------------------
  211|    598|            throw StaticError(ast->location, "Unknown variable: " + encode_utf8(ast->id->name));
  212|    598|        }
  213|  72.3M|        r.insert(ast->id);
  214|  72.3M|    } break;
  215|      0|    default:
  ------------------
  |  Branch (215:5): [True: 0, False: 238M]
  ------------------
  216|      0|        std::cerr << "INTERNAL ERROR: Unknown AST: " << ast_ << std::endl;
  217|      0|        std::abort();
  218|      0|        break;
  219|   238M|    }
  220|       |
  221|   238M|    for (auto *id : r)
  ------------------
  |  Branch (221:19): [True: 324M, False: 238M]
  ------------------
  222|   324M|        ast_->freeVariables.push_back(id);
  223|       |
  224|   238M|    return r;
  225|   238M|}
static_analysis.cpp:_ZN7jsonnet8internalL6appendERNSt3__13setIPKNS0_10IdentifierENS1_4lessIS5_EENS1_9allocatorIS5_EEEERKSA_:
   29|   260M|{
   30|   260M|    r.insert(s.begin(), s.end());
   31|   260M|}

libjsonnet.cpp:_ZN7jsonnet8internallsERNSt3__113basic_ostreamIcNS1_11char_traitsIcEEEERKNS0_11StaticErrorE:
  108|  3.96k|{
  109|  3.96k|    o << err.toString();
  110|  3.96k|    return o;
  111|  3.96k|}
_ZNK7jsonnet8internal11StaticError8toStringEv:
   97|  3.96k|    {
   98|  3.96k|        std::stringstream ss;
   99|  3.96k|        if (location.isSet()) {
  ------------------
  |  Branch (99:13): [True: 3.96k, False: 6]
  ------------------
  100|  3.96k|            ss << location << ":";
  101|  3.96k|        }
  102|  3.96k|        ss << " " << msg;
  103|  3.96k|        return ss.str();
  104|  3.96k|    }
_ZNK7jsonnet8internal13LocationRange5isSetEv:
   58|   321k|    {
   59|   321k|        return begin.isSet();
   60|   321k|    }
_ZNK7jsonnet8internal8Location5isSetEv:
   31|   321k|    {
   32|   321k|        return line != 0;
   33|   321k|    }
libjsonnet.cpp:_ZN7jsonnet8internallsERNSt3__113basic_ostreamIcNS1_11char_traitsIcEEEERKNS0_13LocationRangeE:
   64|  22.8k|{
   65|  22.8k|    if (loc.file.length() > 0)
  ------------------
  |  Branch (65:9): [True: 9.06k, False: 13.7k]
  ------------------
   66|  9.06k|        o << loc.file;
   67|  22.8k|    if (loc.isSet()) {
  ------------------
  |  Branch (67:9): [True: 21.5k, False: 1.32k]
  ------------------
   68|  21.5k|        if (loc.file.length() > 0)
  ------------------
  |  Branch (68:13): [True: 7.82k, False: 13.6k]
  ------------------
   69|  7.82k|            o << ":";
   70|  21.5k|        if (loc.begin.line == loc.end.line) {
  ------------------
  |  Branch (70:13): [True: 18.2k, False: 3.23k]
  ------------------
   71|  18.2k|            if (loc.begin.column == loc.end.column - 1) {
  ------------------
  |  Branch (71:17): [True: 4.09k, False: 14.1k]
  ------------------
   72|  4.09k|                o << loc.begin;
   73|  14.1k|            } else {
   74|  14.1k|                o << loc.begin << "-" << loc.end.column;
   75|  14.1k|            }
   76|  18.2k|        } else {
   77|  3.23k|            o << "(" << loc.begin << ")-(" << loc.end << ")";
   78|  3.23k|        }
   79|  21.5k|    }
   80|  22.8k|    return o;
   81|  22.8k|}
libjsonnet.cpp:_ZN7jsonnet8internallsERNSt3__113basic_ostreamIcNS1_11char_traitsIcEEEERKNS0_8LocationE:
   41|  24.7k|{
   42|  24.7k|    o << loc.line << ":" << loc.column;
   43|  24.7k|    return o;
   44|  24.7k|}
_ZN7jsonnet8internal8LocationC2Ev:
   28|  4.49M|    Location(void) : line(0), column(0) {}
_ZN7jsonnet8internal8LocationC2Emm:
   29|   208M|    Location(unsigned long line, unsigned long column) : line(line), column(column) {}
_ZNK7jsonnet8internal8Location9successorEv:
   35|    521|    {
   36|    521|        return Location(this->line, this->column + 1);
   37|    521|    }
_ZN7jsonnet8internal13LocationRangeC2Ev:
   50|  2.24M|    LocationRange(void) {}
_ZN7jsonnet8internal13LocationRangeC2ERKNSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEE:
   52|  5.83k|    LocationRange(const std::string &msg) : file(msg) {}
_ZN7jsonnet8internal13LocationRangeC2ERKNSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEERKNS0_8LocationESD_:
   54|   161M|        : file(file), begin(begin), end(end)
   55|   161M|    {
   56|   161M|    }
_ZN7jsonnet8internal11StaticErrorC2ERKNSt3__112basic_stringIcNS2_11char_traitsIcEENS2_9allocatorIcEEEERKNS0_8LocationESA_:
   88|    521|        : location(filename, location, location.successor()), msg(msg)
   89|    521|    {
   90|    521|    }
_ZN7jsonnet8internal11StaticErrorC2ERKNS0_13LocationRangeERKNSt3__112basic_stringIcNS5_11char_traitsIcEENS5_9allocatorIcEEEE:
   92|  3.44k|        : location(location), msg(msg)
   93|  3.44k|    {
   94|  3.44k|    }

_ZN7jsonnet8internal22jsonnet_string_unparseERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEEb:
   25|   525k|{
   26|   525k|    UStringStream ss;
   27|   525k|    ss << (single ? U'\'' : U'\"');
  ------------------
  |  Branch (27:12): [True: 0, False: 525k]
  ------------------
   28|   525k|    ss << jsonnet_string_escape(str, single);
   29|   525k|    ss << (single ? U'\'' : U'\"');
  ------------------
  |  Branch (29:12): [True: 0, False: 525k]
  ------------------
   30|   525k|    return ss.str();
   31|   525k|}
_ZN7jsonnet8internal21jsonnet_string_escapeERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEEb:
   34|   525k|{
   35|   525k|    UStringStream ss;
   36|   987M|    for (std::size_t i = 0; i < str.length(); ++i) {
  ------------------
  |  Branch (36:29): [True: 987M, False: 525k]
  ------------------
   37|   987M|        char32_t c = str[i];
   38|   987M|        switch (c) {
   39|   519k|            case U'\"': ss << (single ? U"\"" : U"\\\""); break;
  ------------------
  |  Branch (39:13): [True: 519k, False: 986M]
  |  Branch (39:32): [True: 0, False: 519k]
  ------------------
   40|   363k|            case U'\'': ss << (single ? U"\\\'" : U"\'"); break;
  ------------------
  |  Branch (40:13): [True: 363k, False: 986M]
  |  Branch (40:32): [True: 0, False: 363k]
  ------------------
   41|   939M|            case U'\\': ss << U"\\\\"; break;
  ------------------
  |  Branch (41:13): [True: 939M, False: 47.4M]
  ------------------
   42|  37.6k|            case U'\b': ss << U"\\b"; break;
  ------------------
  |  Branch (42:13): [True: 37.6k, False: 987M]
  ------------------
   43|  1.48M|            case U'\f': ss << U"\\f"; break;
  ------------------
  |  Branch (43:13): [True: 1.48M, False: 985M]
  ------------------
   44|  7.95M|            case U'\n': ss << U"\\n"; break;
  ------------------
  |  Branch (44:13): [True: 7.95M, False: 979M]
  ------------------
   45|   433k|            case U'\r': ss << U"\\r"; break;
  ------------------
  |  Branch (45:13): [True: 433k, False: 986M]
  ------------------
   46|   775k|            case U'\t': ss << U"\\t"; break;
  ------------------
  |  Branch (46:13): [True: 775k, False: 986M]
  ------------------
   47|    453|            case U'\0': ss << U"\\u0000"; break;
  ------------------
  |  Branch (47:13): [True: 453, False: 987M]
  ------------------
   48|  35.8M|            default: {
  ------------------
  |  Branch (48:13): [True: 35.8M, False: 951M]
  ------------------
   49|  35.8M|                if (c < 0x20 || (c >= 0x7f && c <= 0x9f)) {
  ------------------
  |  Branch (49:21): [True: 9.77M, False: 26.0M]
  |  Branch (49:34): [True: 6.88M, False: 19.2M]
  |  Branch (49:47): [True: 772k, False: 6.11M]
  ------------------
   50|       |                    // Unprintable, use \u
   51|  10.5M|                    std::stringstream ss8;
   52|  10.5M|                    ss8 << "\\u" << std::hex << std::setfill('0') << std::setw(4)
   53|  10.5M|                        << (unsigned long)(c);
   54|  10.5M|                    ss << decode_utf8(ss8.str());
   55|  25.3M|                } else {
   56|       |                    // Printable, write verbatim
   57|  25.3M|                    ss << c;
   58|  25.3M|                }
   59|  35.8M|            }
   60|   987M|        }
   61|   987M|    }
   62|   525k|    return ss.str();
   63|   525k|}
_ZN7jsonnet8internal28jsonnet_string_parse_unicodeERKNS0_13LocationRangeEPKDi:
   66|  19.2k|{
   67|  19.2k|    unsigned long codepoint = 0;
   68|       |    // Expect 4 hex digits.
   69|  96.2k|    for (unsigned i = 0; i < 4; ++i) {
  ------------------
  |  Branch (69:26): [True: 77.0k, False: 19.2k]
  ------------------
   70|  77.0k|        auto x = (unsigned char)(c[i]);
   71|  77.0k|        unsigned digit;
   72|  77.0k|        if (x == '\0') {
  ------------------
  |  Branch (72:13): [True: 22, False: 77.0k]
  ------------------
   73|     22|            auto msg = "Truncated unicode escape sequence in string literal.";
   74|     22|            throw StaticError(loc, msg);
   75|  77.0k|        } else if (x >= '0' && x <= '9') {
  ------------------
  |  Branch (75:20): [True: 77.0k, False: 11]
  |  Branch (75:32): [True: 56.5k, False: 20.4k]
  ------------------
   76|  56.5k|            digit = x - '0';
   77|  56.5k|        } else if (x >= 'a' && x <= 'f') {
  ------------------
  |  Branch (77:20): [True: 7.65k, False: 12.8k]
  |  Branch (77:32): [True: 7.64k, False: 11]
  ------------------
   78|  7.64k|            digit = x - 'a' + 10;
   79|  12.8k|        } else if (x >= 'A' && x <= 'F') {
  ------------------
  |  Branch (79:20): [True: 12.8k, False: 16]
  |  Branch (79:32): [True: 12.8k, False: 22]
  ------------------
   80|  12.8k|            digit = x - 'A' + 10;
   81|  12.8k|        } else {
   82|     38|            std::stringstream ss;
   83|     38|            ss << "Malformed unicode escape character, "
   84|     38|               << "should be hex: '" << x << "'";
   85|     38|            throw StaticError(loc, ss.str());
   86|     38|        }
   87|  76.9k|        codepoint *= 16;
   88|  76.9k|        codepoint += digit;
   89|  76.9k|    }
   90|  19.2k|    return codepoint;
   91|  19.2k|}
_ZN7jsonnet8internal16is_bmp_codepointEm:
   94|  18.7k|{
   95|  18.7k|    return codepoint < 0xd800 || (codepoint >= 0xe000 && codepoint < 0x10000);
  ------------------
  |  Branch (95:12): [True: 15.7k, False: 2.94k]
  |  Branch (95:35): [True: 2.39k, False: 552]
  |  Branch (95:58): [True: 2.39k, False: 0]
  ------------------
   96|  18.7k|}
_ZN7jsonnet8internal23decode_utf16_surrogatesERKNS0_13LocationRangeEmm:
   99|    529|{
  100|    529|    if (high >= 0xd800 && high < 0xdc00 && low >= 0xdc00 && low < 0xe000) {
  ------------------
  |  Branch (100:9): [True: 529, False: 0]
  |  Branch (100:27): [True: 527, False: 2]
  |  Branch (100:44): [True: 519, False: 8]
  |  Branch (100:61): [True: 518, False: 1]
  ------------------
  101|    518|        return 0x10000 + ((high & 0x03ff) << 10) + (low & 0x03ff);
  102|    518|    } else {
  103|     11|        std::stringstream ss;
  104|     11|        ss << "Invalid UTF-16 bytes";
  105|     11|        throw StaticError(loc, ss.str());
  106|     11|    }
  107|    529|}
_ZN7jsonnet8internal23jsonnet_string_unescapeERKNS0_13LocationRangeERKNSt3__112basic_stringIDiNS4_11char_traitsIDiEENS4_9allocatorIDiEEEE:
  110|  3.66M|{
  111|  3.66M|    UString r;
  112|  3.66M|    const char32_t *s_ptr = s.c_str();
  113|  65.5M|    for (const char32_t *c = s_ptr; *c != U'\0'; ++c) {
  ------------------
  |  Branch (113:37): [True: 61.8M, False: 3.66M]
  ------------------
  114|  61.8M|        switch (*c) {
  115|   443k|            case '\\':
  ------------------
  |  Branch (115:13): [True: 443k, False: 61.4M]
  ------------------
  116|   443k|                switch (*(++c)) {
  117|  15.7k|                    case '"':
  ------------------
  |  Branch (117:21): [True: 15.7k, False: 428k]
  ------------------
  118|  17.8k|                    case '\'': r += *c; break;
  ------------------
  |  Branch (118:21): [True: 2.18k, False: 441k]
  ------------------
  119|       |
  120|  85.5k|                    case '\\': r += *c; break;
  ------------------
  |  Branch (120:21): [True: 85.5k, False: 358k]
  ------------------
  121|       |
  122|  6.27k|                    case '/': r += *c; break;
  ------------------
  |  Branch (122:21): [True: 6.27k, False: 437k]
  ------------------
  123|       |
  124|  9.10k|                    case 'b': r += '\b'; break;
  ------------------
  |  Branch (124:21): [True: 9.10k, False: 434k]
  ------------------
  125|       |
  126|  29.0k|                    case 'f': r += '\f'; break;
  ------------------
  |  Branch (126:21): [True: 29.0k, False: 414k]
  ------------------
  127|       |
  128|   212k|                    case 'n': r += '\n'; break;
  ------------------
  |  Branch (128:21): [True: 212k, False: 231k]
  ------------------
  129|       |
  130|  15.9k|                    case 'r': r += '\r'; break;
  ------------------
  |  Branch (130:21): [True: 15.9k, False: 427k]
  ------------------
  131|       |
  132|  48.9k|                    case 't': r += '\t'; break;
  ------------------
  |  Branch (132:21): [True: 48.9k, False: 394k]
  ------------------
  133|       |
  134|  18.7k|                    case 'u': {
  ------------------
  |  Branch (134:21): [True: 18.7k, False: 425k]
  ------------------
  135|  18.7k|                        ++c;  // Consume the 'u'.
  136|  18.7k|                        unsigned long codepoint = jsonnet_string_parse_unicode(loc, c);
  137|       |
  138|       |                        // Leave us on the last char, ready for the ++c at
  139|       |                        // the outer for loop.
  140|  18.7k|                        c += 3;
  141|  18.7k|                        if (!is_bmp_codepoint(codepoint)) {
  ------------------
  |  Branch (141:29): [True: 552, False: 18.2k]
  ------------------
  142|    552|                           if (*(++c) != '\\') {
  ------------------
  |  Branch (142:32): [True: 13, False: 539]
  ------------------
  143|     13|                                std::stringstream ss;
  144|     13|                                ss << "Invalid non-BMP Unicode escape in string literal";
  145|     13|                                throw StaticError(loc, ss.str());
  146|     13|                           }
  147|    539|                           if (*(++c) != 'u') {
  ------------------
  |  Branch (147:32): [True: 5, False: 534]
  ------------------
  148|      5|                                std::stringstream ss;
  149|      5|                                ss << "Invalid non-BMP Unicode escape in string literal";
  150|      5|                                throw StaticError(loc, ss.str());
  151|      5|                           }
  152|    534|                           ++c;
  153|    534|                           unsigned long codepoint2 = jsonnet_string_parse_unicode(loc, c);
  154|    534|                           c += 3;
  155|    534|                           codepoint = decode_utf16_surrogates(loc, codepoint, codepoint2);
  156|    534|                       }
  157|  18.7k|                       r += codepoint;
  158|  18.7k|                    } break;
  159|       |
  160|      8|                    case '\0': {
  ------------------
  |  Branch (160:21): [True: 8, False: 443k]
  ------------------
  161|      8|                        auto msg = "Truncated escape sequence in string literal.";
  162|      8|                        throw StaticError(loc, msg);
  163|  18.7k|                    }
  164|       |
  165|    126|                    default: {
  ------------------
  |  Branch (165:21): [True: 126, False: 443k]
  ------------------
  166|    126|                        std::stringstream ss;
  167|    126|                        std::string utf8;
  168|    126|                        encode_utf8(*c, utf8);
  169|    126|                        ss << "Unknown escape sequence in string literal: '" << utf8 << "'";
  170|    126|                        throw StaticError(loc, ss.str());
  171|  18.7k|                    }
  172|   443k|                }
  173|   443k|                break;
  174|       |
  175|  61.4M|            default:
  ------------------
  |  Branch (175:13): [True: 61.4M, False: 443k]
  ------------------
  176|       |                // Just a regular letter.
  177|  61.4M|                r += *c;
  178|  61.8M|        }
  179|  61.8M|    }
  180|  3.66M|    return r;
  181|  3.66M|}

_ZN7jsonnet8internal13UStringStreamlsERKNSt3__112basic_stringIDiNS2_11char_traitsIDiEENS2_9allocatorIDiEEEE:
  155|  17.0M|    {
  156|  17.0M|        buf.append(s);
  157|  17.0M|        return *this;
  158|  17.0M|    }
_ZN7jsonnet8internal13UStringStreamlsEPKDi:
  160|   955M|    {
  161|   955M|        buf.append(s);
  162|   955M|        return *this;
  163|   955M|    }
_ZN7jsonnet8internal13UStringStreamlsEDi:
  165|  26.3M|    {
  166|  26.3M|        buf.push_back(c);
  167|  26.3M|        return *this;
  168|  26.3M|    }
_ZN7jsonnet8internal13UStringStream3strEv:
  179|  4.44M|    {
  180|  4.44M|        return buf;
  181|  4.44M|    }
parser.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
  140|  36.7M|{
  141|  36.7M|    UString r;
  142|   251M|    for (size_t i = 0; i < s.length(); ++i)
  ------------------
  |  Branch (142:24): [True: 214M, False: 36.7M]
  ------------------
  143|   214M|        r.push_back(decode_utf8(s, i));
  144|  36.7M|    return r;
  145|  36.7M|}
parser.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERm:
   75|   214M|{
   76|   214M|    char c0 = str[i];
   77|   214M|    if ((c0 & 0x80) == 0) {  // 0xxxxxxx
  ------------------
  |  Branch (77:9): [True: 210M, False: 4.27M]
  ------------------
   78|   210M|        return c0;
   79|   210M|    } else if ((c0 & 0xE0) == 0xC0) {  // 110yyyxx 10xxxxxx
  ------------------
  |  Branch (79:16): [True: 401k, False: 3.87M]
  ------------------
   80|   401k|        if (i + 1 >= str.length()) {
  ------------------
  |  Branch (80:13): [True: 4.13k, False: 397k]
  ------------------
   81|  4.13k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  4.13k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   82|  4.13k|        }
   83|   397k|        char c1 = str[++i];
   84|   397k|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (84:13): [True: 304k, False: 92.9k]
  ------------------
   85|   304k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|   304k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   86|   304k|        }
   87|  92.9k|        return ((c0 & 0x1F) << 6ul) | (c1 & 0x3F);
   88|  3.87M|    } else if ((c0 & 0xF0) == 0xE0) {  // 1110yyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (88:16): [True: 242k, False: 3.63M]
  ------------------
   89|   242k|        if (i + 2 >= str.length()) {
  ------------------
  |  Branch (89:13): [True: 2.65k, False: 239k]
  ------------------
   90|  2.65k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  2.65k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   91|  2.65k|        }
   92|   239k|        char c1 = str[++i];
   93|   239k|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (93:13): [True: 192k, False: 46.8k]
  ------------------
   94|   192k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|   192k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   95|   192k|        }
   96|  46.8k|        char c2 = str[++i];
   97|  46.8k|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (97:13): [True: 9.21k, False: 37.6k]
  ------------------
   98|  9.21k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  9.21k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   99|  9.21k|        }
  100|  37.6k|        return ((c0 & 0xF) << 12ul) | ((c1 & 0x3F) << 6) | (c2 & 0x3F);
  101|  3.63M|    } else if ((c0 & 0xF8) == 0xF0) {  // 11110zzz 10zzyyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (101:16): [True: 78.7k, False: 3.55M]
  ------------------
  102|  78.7k|        if (i + 3 >= str.length()) {
  ------------------
  |  Branch (102:13): [True: 3.19k, False: 75.5k]
  ------------------
  103|  3.19k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  3.19k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  104|  3.19k|        }
  105|  75.5k|        char c1 = str[++i];
  106|  75.5k|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (106:13): [True: 44.9k, False: 30.5k]
  ------------------
  107|  44.9k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  44.9k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  108|  44.9k|        }
  109|  30.5k|        char c2 = str[++i];
  110|  30.5k|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (110:13): [True: 2.97k, False: 27.5k]
  ------------------
  111|  2.97k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  2.97k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  112|  2.97k|        }
  113|  27.5k|        char c3 = str[++i];
  114|  27.5k|        if ((c3 & 0xC0) != 0x80) {
  ------------------
  |  Branch (114:13): [True: 18.2k, False: 9.25k]
  ------------------
  115|  18.2k|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  18.2k|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  116|  18.2k|        }
  117|  9.25k|        return ((c0 & 0x7) << 18ul) | ((c1 & 0x3F) << 12ul) | ((c2 & 0x3F) << 6) | (c3 & 0x3F);
  118|  3.55M|    } else {
  119|  3.55M|        return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|  3.55M|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  120|  3.55M|    }
  121|   214M|}
static_analysis.cpp:_ZN7jsonnet8internalL11encode_utf8ERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEE:
  133|    603|{
  134|    603|    std::string r;
  135|    603|    encode_utf8(s, r);
  136|    603|    return r;
  137|    603|}
static_analysis.cpp:_ZN7jsonnet8internalL11encode_utf8ERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEERNS2_IcNS3_IcEENS5_IcEEEE:
  127|    603|{
  128|    603|    for (char32_t cp : s)
  ------------------
  |  Branch (128:22): [True: 9.81M, False: 603]
  ------------------
  129|  9.81M|        encode_utf8(cp, r);
  130|    603|}
static_analysis.cpp:_ZN7jsonnet8internalL11encode_utf8EDiRNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
   33|  9.81M|{
   34|  9.81M|    if (x >= JSONNET_CODEPOINT_MAX)
  ------------------
  |  |   22|  9.81M|#define JSONNET_CODEPOINT_MAX 0x110000
  ------------------
  |  Branch (34:9): [True: 0, False: 9.81M]
  ------------------
   35|      0|        x = JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   36|       |
   37|       |    // 00ZZZzzz 00zzYYYY 00Yyyyxx 00xxxxxx
   38|  9.81M|    long bytes = ((x & 0x1C0000) << 6) | ((x & 0x03F000) << 4) | ((x & 0x0FC0) << 2) | (x & 0x3F);
   39|       |
   40|  9.81M|    if (x < 0x80) {
  ------------------
  |  Branch (40:9): [True: 9.81M, False: 0]
  ------------------
   41|  9.81M|        s.push_back((char)x);
   42|  9.81M|        return 1;
   43|  9.81M|    } else if (x < 0x800) {  // note that capital 'Y' bits must be 0
  ------------------
  |  Branch (43:16): [True: 0, False: 0]
  ------------------
   44|      0|        bytes |= 0xC080;
   45|      0|        s.push_back((bytes >> 8) & 0xFF);
   46|      0|        s.push_back((bytes >> 0) & 0xFF);
   47|      0|        return 2;
   48|      0|    } else if (x < 0x10000) {  // note that 'z' bits must be 0
  ------------------
  |  Branch (48:16): [True: 0, False: 0]
  ------------------
   49|      0|        bytes |= 0xE08080;
   50|      0|        s.push_back((bytes >> 16) & 0xFF);
   51|      0|        s.push_back((bytes >> 8) & 0xFF);
   52|      0|        s.push_back((bytes >> 0) & 0xFF);
   53|      0|        return 3;
   54|      0|    } else if (x < 0x110000) {  // note that capital 'Z' bits must be 0
  ------------------
  |  Branch (54:16): [True: 0, False: 0]
  ------------------
   55|      0|        bytes |= 0xF0808080;
   56|      0|        s.push_back((bytes >> 24) & 0xFF);
   57|      0|        s.push_back((bytes >> 16) & 0xFF);
   58|      0|        s.push_back((bytes >> 8) & 0xFF);
   59|      0|        s.push_back((bytes >> 0) & 0xFF);
   60|      0|        return 4;
   61|      0|    } else {
   62|      0|        std::cerr << "Should never get here." << std::endl;
   63|      0|        abort();
   64|      0|    }
   65|  9.81M|}
string_utils.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
  140|  10.5M|{
  141|  10.5M|    UString r;
  142|  73.8M|    for (size_t i = 0; i < s.length(); ++i)
  ------------------
  |  Branch (142:24): [True: 63.2M, False: 10.5M]
  ------------------
  143|  63.2M|        r.push_back(decode_utf8(s, i));
  144|  10.5M|    return r;
  145|  10.5M|}
string_utils.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERm:
   75|  63.2M|{
   76|  63.2M|    char c0 = str[i];
   77|  63.2M|    if ((c0 & 0x80) == 0) {  // 0xxxxxxx
  ------------------
  |  Branch (77:9): [True: 63.2M, False: 0]
  ------------------
   78|  63.2M|        return c0;
   79|  63.2M|    } else if ((c0 & 0xE0) == 0xC0) {  // 110yyyxx 10xxxxxx
  ------------------
  |  Branch (79:16): [True: 0, False: 0]
  ------------------
   80|      0|        if (i + 1 >= str.length()) {
  ------------------
  |  Branch (80:13): [True: 0, False: 0]
  ------------------
   81|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   82|      0|        }
   83|      0|        char c1 = str[++i];
   84|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (84:13): [True: 0, False: 0]
  ------------------
   85|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   86|      0|        }
   87|      0|        return ((c0 & 0x1F) << 6ul) | (c1 & 0x3F);
   88|      0|    } else if ((c0 & 0xF0) == 0xE0) {  // 1110yyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (88:16): [True: 0, False: 0]
  ------------------
   89|      0|        if (i + 2 >= str.length()) {
  ------------------
  |  Branch (89:13): [True: 0, False: 0]
  ------------------
   90|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   91|      0|        }
   92|      0|        char c1 = str[++i];
   93|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (93:13): [True: 0, False: 0]
  ------------------
   94|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   95|      0|        }
   96|      0|        char c2 = str[++i];
   97|      0|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (97:13): [True: 0, False: 0]
  ------------------
   98|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   99|      0|        }
  100|      0|        return ((c0 & 0xF) << 12ul) | ((c1 & 0x3F) << 6) | (c2 & 0x3F);
  101|      0|    } else if ((c0 & 0xF8) == 0xF0) {  // 11110zzz 10zzyyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (101:16): [True: 0, False: 0]
  ------------------
  102|      0|        if (i + 3 >= str.length()) {
  ------------------
  |  Branch (102:13): [True: 0, False: 0]
  ------------------
  103|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  104|      0|        }
  105|      0|        char c1 = str[++i];
  106|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (106:13): [True: 0, False: 0]
  ------------------
  107|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  108|      0|        }
  109|      0|        char c2 = str[++i];
  110|      0|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (110:13): [True: 0, False: 0]
  ------------------
  111|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  112|      0|        }
  113|      0|        char c3 = str[++i];
  114|      0|        if ((c3 & 0xC0) != 0x80) {
  ------------------
  |  Branch (114:13): [True: 0, False: 0]
  ------------------
  115|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  116|      0|        }
  117|      0|        return ((c0 & 0x7) << 18ul) | ((c1 & 0x3F) << 12ul) | ((c2 & 0x3F) << 6) | (c3 & 0x3F);
  118|      0|    } else {
  119|      0|        return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  120|      0|    }
  121|  63.2M|}
string_utils.cpp:_ZN7jsonnet8internalL11encode_utf8EDiRNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
   33|    126|{
   34|    126|    if (x >= JSONNET_CODEPOINT_MAX)
  ------------------
  |  |   22|    126|#define JSONNET_CODEPOINT_MAX 0x110000
  ------------------
  |  Branch (34:9): [True: 1, False: 125]
  ------------------
   35|      1|        x = JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      1|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   36|       |
   37|       |    // 00ZZZzzz 00zzYYYY 00Yyyyxx 00xxxxxx
   38|    126|    long bytes = ((x & 0x1C0000) << 6) | ((x & 0x03F000) << 4) | ((x & 0x0FC0) << 2) | (x & 0x3F);
   39|       |
   40|    126|    if (x < 0x80) {
  ------------------
  |  Branch (40:9): [True: 71, False: 55]
  ------------------
   41|     71|        s.push_back((char)x);
   42|     71|        return 1;
   43|     71|    } else if (x < 0x800) {  // note that capital 'Y' bits must be 0
  ------------------
  |  Branch (43:16): [True: 8, False: 47]
  ------------------
   44|      8|        bytes |= 0xC080;
   45|      8|        s.push_back((bytes >> 8) & 0xFF);
   46|      8|        s.push_back((bytes >> 0) & 0xFF);
   47|      8|        return 2;
   48|     47|    } else if (x < 0x10000) {  // note that 'z' bits must be 0
  ------------------
  |  Branch (48:16): [True: 37, False: 10]
  ------------------
   49|     37|        bytes |= 0xE08080;
   50|     37|        s.push_back((bytes >> 16) & 0xFF);
   51|     37|        s.push_back((bytes >> 8) & 0xFF);
   52|     37|        s.push_back((bytes >> 0) & 0xFF);
   53|     37|        return 3;
   54|     37|    } else if (x < 0x110000) {  // note that capital 'Z' bits must be 0
  ------------------
  |  Branch (54:16): [True: 10, False: 0]
  ------------------
   55|     10|        bytes |= 0xF0808080;
   56|     10|        s.push_back((bytes >> 24) & 0xFF);
   57|     10|        s.push_back((bytes >> 16) & 0xFF);
   58|     10|        s.push_back((bytes >> 8) & 0xFF);
   59|     10|        s.push_back((bytes >> 0) & 0xFF);
   60|     10|        return 4;
   61|     10|    } else {
   62|      0|        std::cerr << "Should never get here." << std::endl;
   63|      0|        abort();
   64|      0|    }
   65|    126|}
vm.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
  140|  1.73M|{
  141|  1.73M|    UString r;
  142|  4.45M|    for (size_t i = 0; i < s.length(); ++i)
  ------------------
  |  Branch (142:24): [True: 2.72M, False: 1.73M]
  ------------------
  143|  2.72M|        r.push_back(decode_utf8(s, i));
  144|  1.73M|    return r;
  145|  1.73M|}
vm.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERm:
   75|  2.72M|{
   76|  2.72M|    char c0 = str[i];
   77|  2.72M|    if ((c0 & 0x80) == 0) {  // 0xxxxxxx
  ------------------
  |  Branch (77:9): [True: 2.72M, False: 0]
  ------------------
   78|  2.72M|        return c0;
   79|  2.72M|    } else if ((c0 & 0xE0) == 0xC0) {  // 110yyyxx 10xxxxxx
  ------------------
  |  Branch (79:16): [True: 0, False: 0]
  ------------------
   80|      0|        if (i + 1 >= str.length()) {
  ------------------
  |  Branch (80:13): [True: 0, False: 0]
  ------------------
   81|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   82|      0|        }
   83|      0|        char c1 = str[++i];
   84|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (84:13): [True: 0, False: 0]
  ------------------
   85|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   86|      0|        }
   87|      0|        return ((c0 & 0x1F) << 6ul) | (c1 & 0x3F);
   88|      0|    } else if ((c0 & 0xF0) == 0xE0) {  // 1110yyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (88:16): [True: 0, False: 0]
  ------------------
   89|      0|        if (i + 2 >= str.length()) {
  ------------------
  |  Branch (89:13): [True: 0, False: 0]
  ------------------
   90|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   91|      0|        }
   92|      0|        char c1 = str[++i];
   93|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (93:13): [True: 0, False: 0]
  ------------------
   94|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   95|      0|        }
   96|      0|        char c2 = str[++i];
   97|      0|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (97:13): [True: 0, False: 0]
  ------------------
   98|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   99|      0|        }
  100|      0|        return ((c0 & 0xF) << 12ul) | ((c1 & 0x3F) << 6) | (c2 & 0x3F);
  101|      0|    } else if ((c0 & 0xF8) == 0xF0) {  // 11110zzz 10zzyyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (101:16): [True: 0, False: 0]
  ------------------
  102|      0|        if (i + 3 >= str.length()) {
  ------------------
  |  Branch (102:13): [True: 0, False: 0]
  ------------------
  103|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  104|      0|        }
  105|      0|        char c1 = str[++i];
  106|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (106:13): [True: 0, False: 0]
  ------------------
  107|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  108|      0|        }
  109|      0|        char c2 = str[++i];
  110|      0|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (110:13): [True: 0, False: 0]
  ------------------
  111|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  112|      0|        }
  113|      0|        char c3 = str[++i];
  114|      0|        if ((c3 & 0xC0) != 0x80) {
  ------------------
  |  Branch (114:13): [True: 0, False: 0]
  ------------------
  115|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  116|      0|        }
  117|      0|        return ((c0 & 0x7) << 18ul) | ((c1 & 0x3F) << 12ul) | ((c2 & 0x3F) << 6) | (c3 & 0x3F);
  118|      0|    } else {
  119|      0|        return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  120|      0|    }
  121|  2.72M|}
vm.cpp:_ZN7jsonnet8internalL11encode_utf8ERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEE:
  133|  1.11M|{
  134|  1.11M|    std::string r;
  135|  1.11M|    encode_utf8(s, r);
  136|  1.11M|    return r;
  137|  1.11M|}
vm.cpp:_ZN7jsonnet8internalL11encode_utf8ERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEERNS2_IcNS3_IcEENS5_IcEEEE:
  127|  1.11M|{
  128|  1.11M|    for (char32_t cp : s)
  ------------------
  |  Branch (128:22): [True: 437M, False: 1.11M]
  ------------------
  129|   437M|        encode_utf8(cp, r);
  130|  1.11M|}
vm.cpp:_ZN7jsonnet8internalL11encode_utf8EDiRNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
   33|   437M|{
   34|   437M|    if (x >= JSONNET_CODEPOINT_MAX)
  ------------------
  |  |   22|   437M|#define JSONNET_CODEPOINT_MAX 0x110000
  ------------------
  |  Branch (34:9): [True: 620, False: 437M]
  ------------------
   35|    620|        x = JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|    620|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   36|       |
   37|       |    // 00ZZZzzz 00zzYYYY 00Yyyyxx 00xxxxxx
   38|   437M|    long bytes = ((x & 0x1C0000) << 6) | ((x & 0x03F000) << 4) | ((x & 0x0FC0) << 2) | (x & 0x3F);
   39|       |
   40|   437M|    if (x < 0x80) {
  ------------------
  |  Branch (40:9): [True: 436M, False: 904k]
  ------------------
   41|   436M|        s.push_back((char)x);
   42|   436M|        return 1;
   43|   436M|    } else if (x < 0x800) {  // note that capital 'Y' bits must be 0
  ------------------
  |  Branch (43:16): [True: 79.2k, False: 824k]
  ------------------
   44|  79.2k|        bytes |= 0xC080;
   45|  79.2k|        s.push_back((bytes >> 8) & 0xFF);
   46|  79.2k|        s.push_back((bytes >> 0) & 0xFF);
   47|  79.2k|        return 2;
   48|   824k|    } else if (x < 0x10000) {  // note that 'z' bits must be 0
  ------------------
  |  Branch (48:16): [True: 821k, False: 3.65k]
  ------------------
   49|   821k|        bytes |= 0xE08080;
   50|   821k|        s.push_back((bytes >> 16) & 0xFF);
   51|   821k|        s.push_back((bytes >> 8) & 0xFF);
   52|   821k|        s.push_back((bytes >> 0) & 0xFF);
   53|   821k|        return 3;
   54|   821k|    } else if (x < 0x110000) {  // note that capital 'Z' bits must be 0
  ------------------
  |  Branch (54:16): [True: 3.65k, False: 0]
  ------------------
   55|  3.65k|        bytes |= 0xF0808080;
   56|  3.65k|        s.push_back((bytes >> 24) & 0xFF);
   57|  3.65k|        s.push_back((bytes >> 16) & 0xFF);
   58|  3.65k|        s.push_back((bytes >> 8) & 0xFF);
   59|  3.65k|        s.push_back((bytes >> 0) & 0xFF);
   60|  3.65k|        return 4;
   61|  3.65k|    } else {
   62|      0|        std::cerr << "Should never get here." << std::endl;
   63|      0|        abort();
   64|      0|    }
   65|   437M|}
_ZN7jsonnet8internal13UStringStreamlsIiEERS1_T_:
  171|   738k|    {
  172|   738k|        std::stringstream ss;
  173|   738k|        ss << c;
  174|   738k|        for (char c : ss.str())
  ------------------
  |  Branch (174:21): [True: 824k, False: 738k]
  ------------------
  175|   824k|            buf.push_back(char32_t(c));
  176|   738k|        return *this;
  177|   738k|    }
_ZN7jsonnet8internal13UStringStreamlsIjEERS1_T_:
  171|   222k|    {
  172|   222k|        std::stringstream ss;
  173|   222k|        ss << c;
  174|   222k|        for (char c : ss.str())
  ------------------
  |  Branch (174:21): [True: 776k, False: 222k]
  ------------------
  175|   776k|            buf.push_back(char32_t(c));
  176|   222k|        return *this;
  177|   222k|    }
desugarer.cpp:_ZN7jsonnet8internalL11encode_utf8ERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEE:
  133|   317k|{
  134|   317k|    std::string r;
  135|   317k|    encode_utf8(s, r);
  136|   317k|    return r;
  137|   317k|}
desugarer.cpp:_ZN7jsonnet8internalL11encode_utf8ERKNSt3__112basic_stringIDiNS1_11char_traitsIDiEENS1_9allocatorIDiEEEERNS2_IcNS3_IcEENS5_IcEEEE:
  127|   317k|{
  128|   317k|    for (char32_t cp : s)
  ------------------
  |  Branch (128:22): [True: 2.12M, False: 317k]
  ------------------
  129|  2.12M|        encode_utf8(cp, r);
  130|   317k|}
desugarer.cpp:_ZN7jsonnet8internalL11encode_utf8EDiRNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
   33|  2.12M|{
   34|  2.12M|    if (x >= JSONNET_CODEPOINT_MAX)
  ------------------
  |  |   22|  2.12M|#define JSONNET_CODEPOINT_MAX 0x110000
  ------------------
  |  Branch (34:9): [True: 0, False: 2.12M]
  ------------------
   35|      0|        x = JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   36|       |
   37|       |    // 00ZZZzzz 00zzYYYY 00Yyyyxx 00xxxxxx
   38|  2.12M|    long bytes = ((x & 0x1C0000) << 6) | ((x & 0x03F000) << 4) | ((x & 0x0FC0) << 2) | (x & 0x3F);
   39|       |
   40|  2.12M|    if (x < 0x80) {
  ------------------
  |  Branch (40:9): [True: 2.12M, False: 0]
  ------------------
   41|  2.12M|        s.push_back((char)x);
   42|  2.12M|        return 1;
   43|  2.12M|    } else if (x < 0x800) {  // note that capital 'Y' bits must be 0
  ------------------
  |  Branch (43:16): [True: 0, False: 0]
  ------------------
   44|      0|        bytes |= 0xC080;
   45|      0|        s.push_back((bytes >> 8) & 0xFF);
   46|      0|        s.push_back((bytes >> 0) & 0xFF);
   47|      0|        return 2;
   48|      0|    } else if (x < 0x10000) {  // note that 'z' bits must be 0
  ------------------
  |  Branch (48:16): [True: 0, False: 0]
  ------------------
   49|      0|        bytes |= 0xE08080;
   50|      0|        s.push_back((bytes >> 16) & 0xFF);
   51|      0|        s.push_back((bytes >> 8) & 0xFF);
   52|      0|        s.push_back((bytes >> 0) & 0xFF);
   53|      0|        return 3;
   54|      0|    } else if (x < 0x110000) {  // note that capital 'Z' bits must be 0
  ------------------
  |  Branch (54:16): [True: 0, False: 0]
  ------------------
   55|      0|        bytes |= 0xF0808080;
   56|      0|        s.push_back((bytes >> 24) & 0xFF);
   57|      0|        s.push_back((bytes >> 16) & 0xFF);
   58|      0|        s.push_back((bytes >> 8) & 0xFF);
   59|      0|        s.push_back((bytes >> 0) & 0xFF);
   60|      0|        return 4;
   61|      0|    } else {
   62|      0|        std::cerr << "Should never get here." << std::endl;
   63|      0|        abort();
   64|      0|    }
   65|  2.12M|}
desugarer.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEE:
  140|  7.56k|{
  141|  7.56k|    UString r;
  142|  49.0k|    for (size_t i = 0; i < s.length(); ++i)
  ------------------
  |  Branch (142:24): [True: 41.5k, False: 7.56k]
  ------------------
  143|  41.5k|        r.push_back(decode_utf8(s, i));
  144|  7.56k|    return r;
  145|  7.56k|}
desugarer.cpp:_ZN7jsonnet8internalL11decode_utf8ERKNSt3__112basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERm:
   75|  41.5k|{
   76|  41.5k|    char c0 = str[i];
   77|  41.5k|    if ((c0 & 0x80) == 0) {  // 0xxxxxxx
  ------------------
  |  Branch (77:9): [True: 41.5k, False: 0]
  ------------------
   78|  41.5k|        return c0;
   79|  41.5k|    } else if ((c0 & 0xE0) == 0xC0) {  // 110yyyxx 10xxxxxx
  ------------------
  |  Branch (79:16): [True: 0, False: 0]
  ------------------
   80|      0|        if (i + 1 >= str.length()) {
  ------------------
  |  Branch (80:13): [True: 0, False: 0]
  ------------------
   81|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   82|      0|        }
   83|      0|        char c1 = str[++i];
   84|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (84:13): [True: 0, False: 0]
  ------------------
   85|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   86|      0|        }
   87|      0|        return ((c0 & 0x1F) << 6ul) | (c1 & 0x3F);
   88|      0|    } else if ((c0 & 0xF0) == 0xE0) {  // 1110yyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (88:16): [True: 0, False: 0]
  ------------------
   89|      0|        if (i + 2 >= str.length()) {
  ------------------
  |  Branch (89:13): [True: 0, False: 0]
  ------------------
   90|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   91|      0|        }
   92|      0|        char c1 = str[++i];
   93|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (93:13): [True: 0, False: 0]
  ------------------
   94|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   95|      0|        }
   96|      0|        char c2 = str[++i];
   97|      0|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (97:13): [True: 0, False: 0]
  ------------------
   98|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
   99|      0|        }
  100|      0|        return ((c0 & 0xF) << 12ul) | ((c1 & 0x3F) << 6) | (c2 & 0x3F);
  101|      0|    } else if ((c0 & 0xF8) == 0xF0) {  // 11110zzz 10zzyyyy 10yyyyxx 10xxxxxx
  ------------------
  |  Branch (101:16): [True: 0, False: 0]
  ------------------
  102|      0|        if (i + 3 >= str.length()) {
  ------------------
  |  Branch (102:13): [True: 0, False: 0]
  ------------------
  103|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  104|      0|        }
  105|      0|        char c1 = str[++i];
  106|      0|        if ((c1 & 0xC0) != 0x80) {
  ------------------
  |  Branch (106:13): [True: 0, False: 0]
  ------------------
  107|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  108|      0|        }
  109|      0|        char c2 = str[++i];
  110|      0|        if ((c2 & 0xC0) != 0x80) {
  ------------------
  |  Branch (110:13): [True: 0, False: 0]
  ------------------
  111|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  112|      0|        }
  113|      0|        char c3 = str[++i];
  114|      0|        if ((c3 & 0xC0) != 0x80) {
  ------------------
  |  Branch (114:13): [True: 0, False: 0]
  ------------------
  115|      0|            return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  116|      0|        }
  117|      0|        return ((c0 & 0x7) << 18ul) | ((c1 & 0x3F) << 12ul) | ((c2 & 0x3F) << 6) | (c3 & 0x3F);
  118|      0|    } else {
  119|      0|        return JSONNET_CODEPOINT_ERROR;
  ------------------
  |  |   21|      0|#define JSONNET_CODEPOINT_ERROR 0xfffd
  ------------------
  120|      0|    }
  121|  41.5k|}

_ZN7jsonnet8internal25jsonnet_vm_execute_streamEPNS0_9AllocatorEPKNS0_3ASTERKNSt3__13mapINS6_12basic_stringIcNS6_11char_traitsIcEENS6_9allocatorIcEEEENS0_5VmExtENS6_4lessISD_EENSB_INS6_4pairIKSD_SE_EEEEEEjddRKNS7_ISD_NS0_16VmNativeCallbackESG_NSB_INSH_ISI_SO_EEEEEEPFiPvPKcSW_PPcSY_PmESU_b:
 3539|  3.46k|{
 3540|  3.46k|    Interpreter vm(alloc,
 3541|  3.46k|                   ext_vars,
 3542|  3.46k|                   max_stack,
 3543|  3.46k|                   gc_min_objects,
 3544|  3.46k|                   gc_growth_trigger,
 3545|  3.46k|                   natives,
 3546|  3.46k|                   import_callback,
 3547|  3.46k|                   ctx);
 3548|  3.46k|    vm.evaluate(ast, 0);
 3549|  3.46k|    return vm.manifestStream(string_output);
 3550|  3.46k|}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111InterpreterC2EPNS0_9AllocatorERKNSt3__13mapINS5_12basic_stringIcNS5_11char_traitsIcEENS5_9allocatorIcEEEENS0_5VmExtENS5_4lessISC_EENSA_INS5_4pairIKSC_SD_EEEEEEjddRKNS6_ISC_NS0_16VmNativeCallbackESF_NSA_INSG_ISH_SN_EEEEEEPFiPvPKcSV_PPcSX_PmEST_:
  945|  3.46k|        : heap(gc_min_objects, gc_growth_trigger),
  946|  3.46k|          stack(max_stack),
  947|  3.46k|          alloc(alloc),
  948|  3.46k|          idImport(alloc->makeIdentifier(U"import")),
  949|  3.46k|          idArrayElement(alloc->makeIdentifier(U"array_element")),
  950|  3.46k|          idInvariant(alloc->makeIdentifier(U"object_assert")),
  951|  3.46k|          idInternal(alloc->makeIdentifier(U"__internal__")),
  952|  3.46k|          idJsonObjVar(alloc->makeIdentifier(U"_")),
  953|  3.46k|          idEmpty(alloc->makeIdentifier(U"")),
  954|  3.46k|          jsonObjVar(alloc->make<Var>(LocationRange(), Fodder{}, idJsonObjVar)),
  955|  3.46k|          externalVars(ext_vars),
  956|  3.46k|          importCallback(import_callback),
  957|  3.46k|          importCallbackContext(import_callback_context)
  958|  3.46k|    {
  959|  3.46k|        scratch = makeNull();
  960|       |        // Add a prefix to avoid conflicting with names from `native_callbacks`.
  961|  3.46k|        builtins["std:makeArray"] = &Interpreter::builtinMakeArray;
  962|  3.46k|        builtins["std:pow"] = &Interpreter::builtinPow;
  963|  3.46k|        builtins["std:floor"] = &Interpreter::builtinFloor;
  964|  3.46k|        builtins["std:ceil"] = &Interpreter::builtinCeil;
  965|  3.46k|        builtins["std:sqrt"] = &Interpreter::builtinSqrt;
  966|  3.46k|        builtins["std:sin"] = &Interpreter::builtinSin;
  967|  3.46k|        builtins["std:cos"] = &Interpreter::builtinCos;
  968|  3.46k|        builtins["std:tan"] = &Interpreter::builtinTan;
  969|  3.46k|        builtins["std:asin"] = &Interpreter::builtinAsin;
  970|  3.46k|        builtins["std:acos"] = &Interpreter::builtinAcos;
  971|  3.46k|        builtins["std:atan"] = &Interpreter::builtinAtan;
  972|  3.46k|        builtins["std:type"] = &Interpreter::builtinType;
  973|  3.46k|        builtins["std:filter"] = &Interpreter::builtinFilter;
  974|  3.46k|        builtins["std:objectHasEx"] = &Interpreter::builtinObjectHasEx;
  975|  3.46k|        builtins["std:length"] = &Interpreter::builtinLength;
  976|  3.46k|        builtins["std:objectFieldsEx"] = &Interpreter::builtinObjectFieldsEx;
  977|  3.46k|        builtins["std:objectRemoveKey"] = &Interpreter::builtinObjectRemoveKey;
  978|  3.46k|        builtins["std:codepoint"] = &Interpreter::builtinCodepoint;
  979|  3.46k|        builtins["std:char"] = &Interpreter::builtinChar;
  980|  3.46k|        builtins["std:log"] = &Interpreter::builtinLog;
  981|  3.46k|        builtins["std:exp"] = &Interpreter::builtinExp;
  982|  3.46k|        builtins["std:mantissa"] = &Interpreter::builtinMantissa;
  983|  3.46k|        builtins["std:exponent"] = &Interpreter::builtinExponent;
  984|  3.46k|        builtins["std:modulo"] = &Interpreter::builtinModulo;
  985|  3.46k|        builtins["std:extVar"] = &Interpreter::builtinExtVar;
  986|  3.46k|        builtins["std:primitiveEquals"] = &Interpreter::builtinPrimitiveEquals;
  987|  3.46k|        builtins["std:native"] = &Interpreter::builtinNative;
  988|  3.46k|        builtins["std:md5"] = &Interpreter::builtinMd5;
  989|  3.46k|        builtins["std:trace"] = &Interpreter::builtinTrace;
  990|  3.46k|        builtins["std:splitLimit"] = &Interpreter::builtinSplitLimit;
  991|  3.46k|        builtins["std:substr"] = &Interpreter::builtinSubstr;
  992|  3.46k|        builtins["std:range"] = &Interpreter::builtinRange;
  993|  3.46k|        builtins["std:strReplace"] = &Interpreter::builtinStrReplace;
  994|  3.46k|        builtins["std:asciiLower"] = &Interpreter::builtinAsciiLower;
  995|  3.46k|        builtins["std:asciiUpper"] = &Interpreter::builtinAsciiUpper;
  996|  3.46k|        builtins["std:join"] = &Interpreter::builtinJoin;
  997|  3.46k|        builtins["std:parseJson"] = &Interpreter::builtinParseJson;
  998|  3.46k|        builtins["std:parseYaml"] = &Interpreter::builtinParseYaml;
  999|  3.46k|        builtins["std:encodeUTF8"] = &Interpreter::builtinEncodeUTF8;
 1000|  3.46k|        builtins["std:decodeUTF8"] = &Interpreter::builtinDecodeUTF8;
 1001|  3.46k|        builtins["std:atan2"] = &Interpreter::builtinAtan2;
 1002|  3.46k|        builtins["std:hypot"] = &Interpreter::builtinHypot;
 1003|       |
 1004|       |        // Add a prefix `native:` to names of provided callbacks to ensure they can't clash with the builtins above.
 1005|       |        // Although we have separate lookup tables for them, the HeapClosure and BuiltinFunctionBody types just hold
 1006|       |        // function "name" as a std::string to identify the function.
 1007|  3.46k|        for (const auto& [name, cb] : native_callbacks) {
  ------------------
  |  Branch (1007:37): [True: 0, False: 3.46k]
  ------------------
 1008|      0|            nativeCallbacks.emplace("native:" + name, cb);
 1009|      0|        }
 1010|       |
 1011|  3.46k|        DesugaredObject *stdlib = makeStdlibAST(alloc, "__internal__");
 1012|  3.46k|        jsonnet_static_analysis(stdlib);
 1013|  3.46k|        stdlibAST = stdlib; // stdlibAST is const, so we need to do analysis before this assignment
 1014|  3.46k|        auto stdThunk = makeHeap<HeapThunk>(nullptr, nullptr, 0, static_cast<const AST*>(stdlibAST));
 1015|  3.46k|        stack.newCall(stdThunk->body->location, stdThunk, stdThunk->self, stdThunk->offset, stdThunk->upValues);
 1016|  3.46k|        evaluate(stdThunk->body, 0);
 1017|  3.46k|        stdObject = dynamic_cast<HeapObject*>(scratch.v.h);
 1018|  3.46k|        prepareSourceValThunks();
 1019|  3.46k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15StackC2Ej:
  258|  3.46k|    Stack(unsigned limit) : calls(0), limit(limit) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeNullEv:
  634|   109k|    {
  635|   109k|        Value r;
  636|   109k|        r.t = Value::NULL_TYPE;
  637|   109k|        return r;
  638|   109k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack3topEv:
  291|   420M|    {
  292|   420M|        return stack.back();
  293|   420M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter19validateBuiltinArgsERKNS0_13LocationRangeERKNSt3__112basic_stringIcNS6_11char_traitsIcEENS6_9allocatorIcEEEERKNS6_6vectorINS1_5ValueENSA_ISG_EEEENSF_INSG_4TypeENSA_ISL_EEEE:
 1043|   230k|    {
 1044|   230k|        if (args.size() == params.size()) {
  ------------------
  |  Branch (1044:13): [True: 230k, False: 0]
  ------------------
 1045|   677k|            for (std::size_t i = 0; i < args.size(); ++i) {
  ------------------
  |  Branch (1045:37): [True: 446k, False: 230k]
  ------------------
 1046|   446k|                if (args[i].t != params[i])
  ------------------
  |  Branch (1046:21): [True: 0, False: 446k]
  ------------------
 1047|      0|                    goto bad;
 1048|   446k|            }
 1049|   230k|            return;
 1050|   230k|        }
 1051|      0|    bad:;
 1052|      0|        std::stringstream ss;
 1053|      0|        ss << "Builtin function " + name + " expected (";
 1054|      0|        const char *prefix = "";
 1055|      0|        for (auto p : params) {
  ------------------
  |  Branch (1055:21): [True: 0, False: 0]
  ------------------
 1056|      0|            ss << prefix << type_str(p);
 1057|      0|            prefix = ", ";
 1058|      0|        }
 1059|      0|        ss << ") but got (";
 1060|      0|        prefix = "";
 1061|      0|        for (auto a : args) {
  ------------------
  |  Branch (1061:21): [True: 0, False: 0]
  ------------------
 1062|      0|            ss << prefix << type_str(a);
 1063|      0|            prefix = ", ";
 1064|      0|        }
 1065|      0|        ss << ")";
 1066|      0|        throw makeError(loc, ss.str());
 1067|   230k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter9makeErrorERKNS0_13LocationRangeERKNSt3__112basic_stringIcNS6_11char_traitsIcEENS6_9allocatorIcEEEE:
  567|  2.40k|    {
  568|  2.40k|        return stack.makeError(loc, msg);
  569|  2.40k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack9makeErrorERKNS0_13LocationRangeERKNSt3__112basic_stringIcNS6_11char_traitsIcEENS6_9allocatorIcEEEE:
  368|  2.95k|    {
  369|  2.95k|        std::vector<TraceFrame> stack_trace;
  370|  2.95k|        stack_trace.push_back(TraceFrame(loc));
  371|  1.44M|        for (int i = stack.size() - 1; i >= 0; --i) {
  ------------------
  |  Branch (371:40): [True: 1.43M, False: 2.95k]
  ------------------
  372|  1.43M|            const auto &f = stack[i];
  373|  1.43M|            if (f.isCall()) {
  ------------------
  |  Branch (373:17): [True: 295k, False: 1.14M]
  ------------------
  374|   295k|                if (f.context != nullptr) {
  ------------------
  |  Branch (374:21): [True: 295k, False: 0]
  ------------------
  375|       |                    // Give the last line a name.
  376|   295k|                    stack_trace[stack_trace.size() - 1].name = getName(i, f.context);
  377|   295k|                }
  378|   295k|                if (f.location.isSet() || f.location.file.length() > 0)
  ------------------
  |  Branch (378:21): [True: 232k, False: 62.8k]
  |  Branch (378:43): [True: 101, False: 62.7k]
  ------------------
  379|   232k|                    stack_trace.push_back(TraceFrame(f.location));
  380|   295k|            }
  381|  1.43M|        }
  382|  2.95k|        return RuntimeError(stack_trace, msg);
  383|  2.95k|    }
vm.cpp:_ZNK7jsonnet8internal12_GLOBAL__N_15Frame6isCallEv:
  241|   556M|    {
  242|   556M|        return kind == FRAME_CALL;
  243|   556M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack7getNameEjPKNS1_10HeapEntityE:
  313|   295k|    {
  314|   295k|        std::string name;
  315|  1.42M|        for (int i = from_here - 1; i >= 0; --i) {
  ------------------
  |  Branch (315:37): [True: 1.42M, False: 1.81k]
  ------------------
  316|  1.42M|            const auto &f = stack[i];
  317|  1.42M|            for (const auto &pair : f.bindings) {
  ------------------
  |  Branch (317:35): [True: 490k, False: 1.42M]
  ------------------
  318|   490k|                HeapThunk *thunk = pair.second;
  319|   490k|                if (!thunk->filled)
  ------------------
  |  Branch (319:21): [True: 102k, False: 388k]
  ------------------
  320|   102k|                    continue;
  321|   388k|                if (!thunk->content.isHeap())
  ------------------
  |  Branch (321:21): [True: 52.0k, False: 336k]
  ------------------
  322|  52.0k|                    continue;
  323|   336k|                if (e != thunk->content.v.h)
  ------------------
  |  Branch (323:21): [True: 291k, False: 44.4k]
  ------------------
  324|   291k|                    continue;
  325|  44.4k|                name = encode_utf8(pair.first->name);
  326|  44.4k|            }
  327|       |            // Do not go into the next call frame, keep local reasoning.
  328|  1.42M|            if (f.isCall())
  ------------------
  |  Branch (328:17): [True: 293k, False: 1.13M]
  ------------------
  329|   293k|                break;
  330|  1.42M|        }
  331|       |
  332|   295k|        if (name == "")
  ------------------
  |  Branch (332:13): [True: 250k, False: 44.4k]
  ------------------
  333|   250k|            name = "anonymous";
  334|   295k|        if (dynamic_cast<const HeapObject *>(e)) {
  ------------------
  |  Branch (334:13): [True: 201k, False: 93.4k]
  ------------------
  335|   201k|            return "object <" + name + ">";
  336|   201k|        } else if (auto *thunk = dynamic_cast<const HeapThunk *>(e)) {
  ------------------
  |  Branch (336:26): [True: 43.9k, False: 49.4k]
  ------------------
  337|  43.9k|            if (thunk->name == nullptr) {
  ------------------
  |  Branch (337:17): [True: 7.21k, False: 36.7k]
  ------------------
  338|  7.21k|                return "";  // Argument of builtin, or root (since top level functions).
  339|  36.7k|            } else {
  340|  36.7k|                return "thunk <" + encode_utf8(thunk->name->name) + ">";
  341|  36.7k|            }
  342|  49.4k|        } else {
  343|  49.4k|            const auto *func = static_cast<const HeapClosure *>(e);
  344|  49.4k|            if (func->isBuiltin()) {
  ------------------
  |  Branch (344:17): [True: 0, False: 49.4k]
  ------------------
  345|      0|                return "builtin function <" + func->builtinName + ">";
  346|      0|            }
  347|  49.4k|            return "function <" + name + ">";
  348|  49.4k|        }
  349|   295k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack4markERNS1_4HeapE:
  284|  78.7k|    {
  285|  18.2M|        for (const auto &f : stack) {
  ------------------
  |  Branch (285:28): [True: 18.2M, False: 78.7k]
  ------------------
  286|  18.2M|            f.mark(heap);
  287|  18.2M|        }
  288|  78.7k|    }
vm.cpp:_ZNK7jsonnet8internal12_GLOBAL__N_15Frame4markERNS1_4HeapE:
  225|  18.2M|    {
  226|  18.2M|        heap.markFrom(val);
  227|  18.2M|        heap.markFrom(val2);
  228|  18.2M|        if (context)
  ------------------
  |  Branch (228:13): [True: 12.7M, False: 5.53M]
  ------------------
  229|  12.7M|            heap.markFrom(context);
  230|  18.2M|        if (self)
  ------------------
  |  Branch (230:13): [True: 12.5M, False: 5.76M]
  ------------------
  231|  12.5M|            heap.markFrom(self);
  232|  18.2M|        for (const auto &bind : bindings)
  ------------------
  |  Branch (232:31): [True: 17.9M, False: 18.2M]
  ------------------
  233|  17.9M|            heap.markFrom(bind.second);
  234|  18.2M|        for (const auto &el : elements)
  ------------------
  |  Branch (234:29): [True: 0, False: 18.2M]
  ------------------
  235|      0|            heap.markFrom(el.second);
  236|  18.2M|        for (const auto &th : thunks)
  ------------------
  |  Branch (236:29): [True: 869k, False: 18.2M]
  ------------------
  237|   869k|            heap.markFrom(th);
  238|  18.2M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10makeNumberEd:
  615|  3.30M|    {
  616|  3.30M|        Value r;
  617|  3.30M|        r.t = Value::NUMBER;
  618|  3.30M|        r.v.d = v;
  619|  3.30M|        return r;
  620|  3.30M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter9makeArrayERKNSt3__16vectorIPNS1_9HeapThunkENS3_9allocatorIS6_EEEE:
  641|  1.12M|    {
  642|  1.12M|        Value r;
  643|  1.12M|        r.t = Value::ARRAY;
  644|  1.12M|        r.v.h = makeHeap<HeapArray>(v);
  645|  1.12M|        return r;
  646|  1.12M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapArrayEJRKNSt3__16vectorIPNS1_9HeapThunkENS5_9allocatorIS8_EEEEEEEPT_DpOT0_:
  577|  1.12M|    {
  578|  1.12M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  1.12M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 1.52k, False: 1.11M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  1.52k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  1.52k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  1.52k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  1.52k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 1.52k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|   245k|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 245k, False: 1.52k]
  ------------------
  597|   245k|                heap.markFrom(sourceVal.second);
  598|   245k|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  1.52k|            heap.sweep();
  602|  1.52k|        }
  603|  1.12M|        return r;
  604|  1.12M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10builtinPowERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1103|  1.28k|    {
 1104|  1.28k|        validateBuiltinArgs(loc, "pow", args, {Value::NUMBER, Value::NUMBER});
 1105|  1.28k|        scratch = makeNumberCheck(loc, std::pow(args[0].v.d, args[1].v.d));
 1106|  1.28k|        return nullptr;
 1107|  1.28k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter15makeNumberCheckERKNS0_13LocationRangeEd:
  623|  2.78M|    {
  624|  2.78M|        if (std::isnan(v)) {
  ------------------
  |  Branch (624:13): [True: 0, False: 2.78M]
  ------------------
  625|      0|            throw makeError(loc, "not a number");
  626|      0|        }
  627|  2.78M|        if (std::isinf(v)) {
  ------------------
  |  Branch (627:13): [True: 26, False: 2.78M]
  ------------------
  628|     26|            throw makeError(loc, "overflow");
  629|     26|        }
  630|  2.78M|        return makeNumber(v);
  631|  2.78M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter12builtinFloorERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1110|  13.2k|    {
 1111|  13.2k|        validateBuiltinArgs(loc, "floor", args, {Value::NUMBER});
 1112|  13.2k|        scratch = makeNumberCheck(loc, std::floor(args[0].v.d));
 1113|  13.2k|        return nullptr;
 1114|  13.2k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter11builtinTypeERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1187|  2.28M|    {
 1188|  2.28M|        switch (args[0].t) {
  ------------------
  |  Branch (1188:17): [True: 2.28M, False: 0]
  ------------------
 1189|   157k|            case Value::NULL_TYPE: scratch = makeString(U"null"); return nullptr;
  ------------------
  |  Branch (1189:13): [True: 157k, False: 2.13M]
  ------------------
 1190|       |
 1191|   207k|            case Value::BOOLEAN: scratch = makeString(U"boolean"); return nullptr;
  ------------------
  |  Branch (1191:13): [True: 207k, False: 2.08M]
  ------------------
 1192|       |
 1193|   396k|            case Value::NUMBER: scratch = makeString(U"number"); return nullptr;
  ------------------
  |  Branch (1193:13): [True: 396k, False: 1.89M]
  ------------------
 1194|       |
 1195|   152k|            case Value::ARRAY: scratch = makeString(U"array"); return nullptr;
  ------------------
  |  Branch (1195:13): [True: 152k, False: 2.13M]
  ------------------
 1196|       |
 1197|     19|            case Value::FUNCTION: scratch = makeString(U"function"); return nullptr;
  ------------------
  |  Branch (1197:13): [True: 19, False: 2.28M]
  ------------------
 1198|       |
 1199|   153k|            case Value::OBJECT: scratch = makeString(U"object"); return nullptr;
  ------------------
  |  Branch (1199:13): [True: 153k, False: 2.13M]
  ------------------
 1200|       |
 1201|  1.22M|            case Value::STRING: scratch = makeString(U"string"); return nullptr;
  ------------------
  |  Branch (1201:13): [True: 1.22M, False: 1.06M]
  ------------------
 1202|  2.28M|        }
 1203|      0|        return nullptr;  // Quiet, compiler.
 1204|  2.28M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10makeStringERKNSt3__112basic_stringIDiNS3_11char_traitsIDiEENS3_9allocatorIDiEEEE:
  679|  27.6M|    {
  680|  27.6M|        Value r;
  681|  27.6M|        r.t = Value::STRING;
  682|  27.6M|        r.v.h = makeHeap<HeapString>(v);
  683|  27.6M|        return r;
  684|  27.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_10HeapStringEJRKNSt3__112basic_stringIDiNS5_11char_traitsIDiEENS5_9allocatorIDiEEEEEEEPT_DpOT0_:
  577|  27.6M|    {
  578|  27.6M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  27.6M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 30.9k, False: 27.5M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  30.9k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  30.9k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  30.9k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  30.9k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 30.9k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|  4.98M|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 4.98M, False: 30.9k]
  ------------------
  597|  4.98M|                heap.markFrom(sourceVal.second);
  598|  4.98M|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  30.9k|            heap.sweep();
  602|  30.9k|        }
  603|  27.6M|        return r;
  604|  27.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter12objectFieldsEPKNS1_10HeapObjectEb:
  773|   687k|    {
  774|   687k|        std::set<const Identifier *> r;
  775|   978k|        for (const auto &pair : objectFieldsAux(obj_)) {
  ------------------
  |  Branch (775:31): [True: 978k, False: 687k]
  ------------------
  776|   978k|            if (!manifesting || pair.second != ObjectField::HIDDEN)
  ------------------
  |  Branch (776:17): [True: 0, False: 978k]
  |  Branch (776:33): [True: 943k, False: 34.9k]
  ------------------
  777|   943k|                r.insert(pair.first);
  778|   978k|        }
  779|   687k|        return r;
  780|   687k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter15objectFieldsAuxEPKNS1_10HeapObjectE:
  740|  30.4M|    {
  741|  30.4M|        IdHideMap r;
  742|  30.4M|        if (auto *obj = dynamic_cast<const HeapSimpleObject *>(obj_)) {
  ------------------
  |  Branch (742:19): [True: 15.5M, False: 14.8M]
  ------------------
  743|  15.5M|            for (const auto &f : obj->fields) {
  ------------------
  |  Branch (743:32): [True: 12.4M, False: 15.5M]
  ------------------
  744|  12.4M|                r[f.first] = f.second.hide;
  745|  12.4M|            }
  746|       |
  747|  15.5M|        } else if (auto *obj = dynamic_cast<const HeapRestrictedObject *>(obj_)) {
  ------------------
  |  Branch (747:26): [True: 0, False: 14.8M]
  ------------------
  748|      0|            return obj->retainedKeys;
  749|       |
  750|  14.8M|        } else if (auto *obj = dynamic_cast<const HeapExtendedObject *>(obj_)) {
  ------------------
  |  Branch (750:26): [True: 14.8M, False: 0]
  ------------------
  751|  14.8M|            r = objectFieldsAux(obj->right);
  752|  65.4M|            for (const auto &pair : objectFieldsAux(obj->left)) {
  ------------------
  |  Branch (752:35): [True: 65.4M, False: 14.8M]
  ------------------
  753|  65.4M|                auto it = r.find(pair.first);
  754|  65.4M|                if (it == r.end()) {
  ------------------
  |  Branch (754:21): [True: 54.0M, False: 11.4M]
  ------------------
  755|       |                    // First time it is seen
  756|  54.0M|                    r[pair.first] = pair.second;
  757|  54.0M|                } else if (it->second == ObjectField::INHERIT) {
  ------------------
  |  Branch (757:28): [True: 11.2M, False: 236k]
  ------------------
  758|       |                    // Seen before, but with inherited visibility so use new visibility
  759|  11.2M|                    r[pair.first] = pair.second;
  760|  11.2M|                }
  761|  65.4M|            }
  762|       |
  763|  14.8M|        } else if (auto *obj = dynamic_cast<const HeapComprehensionObject *>(obj_)) {
  ------------------
  |  Branch (763:26): [True: 0, False: 0]
  ------------------
  764|      0|            for (const auto &f : obj->compValues)
  ------------------
  |  Branch (764:32): [True: 0, False: 0]
  ------------------
  765|      0|                r[f.first] = ObjectField::INHERIT;
  766|      0|        }
  767|  30.4M|        return r;
  768|  30.4M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter11makeBooleanEb:
  607|  7.49M|    {
  608|  7.49M|        Value r;
  609|  7.49M|        r.t = Value::BOOLEAN;
  610|  7.49M|        r.v.b = v;
  611|  7.49M|        return r;
  612|  7.49M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter13builtinLengthERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1252|   444k|    {
 1253|   444k|        if (args.size() != 1) {
  ------------------
  |  Branch (1253:13): [True: 0, False: 444k]
  ------------------
 1254|      0|            throw makeError(loc, "length takes 1 parameter.");
 1255|      0|        }
 1256|   444k|        HeapEntity *e = args[0].v.h;
 1257|   444k|        switch (args[0].t) {
 1258|      0|            case Value::OBJECT: {
  ------------------
  |  Branch (1258:13): [True: 0, False: 444k]
  ------------------
 1259|      0|                auto fields = objectFields(static_cast<HeapObject *>(e), true);
 1260|      0|                scratch = makeNumber(fields.size());
 1261|      0|            } break;
 1262|       |
 1263|   237k|            case Value::ARRAY:
  ------------------
  |  Branch (1263:13): [True: 237k, False: 207k]
  ------------------
 1264|   237k|                scratch = makeNumber(static_cast<HeapArray *>(e)->elements.size());
 1265|   237k|                break;
 1266|       |
 1267|   207k|            case Value::STRING:
  ------------------
  |  Branch (1267:13): [True: 207k, False: 237k]
  ------------------
 1268|   207k|                scratch = makeNumber(static_cast<HeapString *>(e)->value.length());
 1269|   207k|                break;
 1270|       |
 1271|      0|            case Value::FUNCTION:
  ------------------
  |  Branch (1271:13): [True: 0, False: 444k]
  ------------------
 1272|      0|                scratch = makeNumber(static_cast<HeapClosure *>(e)->params.size());
 1273|      0|                break;
 1274|       |
 1275|     10|            default:
  ------------------
  |  Branch (1275:13): [True: 10, False: 444k]
  ------------------
 1276|     10|                throw makeError(loc,
 1277|     10|                                "length operates on strings, objects, "
 1278|     10|                                "and arrays, got " +
 1279|     10|                                    type_str(args[0]));
 1280|   444k|        }
 1281|   444k|        return nullptr;
 1282|   444k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter21builtinObjectFieldsExERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1285|   141k|    {
 1286|   141k|        validateBuiltinArgs(loc, "objectFieldsEx", args, {Value::OBJECT, Value::BOOLEAN});
 1287|   141k|        const auto *obj = static_cast<HeapObject *>(args[0].v.h);
 1288|   141k|        bool include_hidden = args[1].v.b;
 1289|       |        // Stash in a set first to sort them.
 1290|   141k|        std::set<UString> fields;
 1291|   272k|        for (const auto &field : objectFields(obj, !include_hidden)) {
  ------------------
  |  Branch (1291:32): [True: 272k, False: 141k]
  ------------------
 1292|   272k|            fields.insert(field->name);
 1293|   272k|        }
 1294|   141k|        scratch = makeArray({});
 1295|   141k|        auto &elements = static_cast<HeapArray *>(scratch.v.h)->elements;
 1296|   272k|        for (const auto &field : fields) {
  ------------------
  |  Branch (1296:32): [True: 272k, False: 141k]
  ------------------
 1297|   272k|            auto *th = makeHeap<HeapThunk>(idArrayElement, nullptr, 0, nullptr);
 1298|   272k|            elements.push_back(th);
 1299|   272k|            th->fill(makeString(field));
 1300|   272k|        }
 1301|   141k|        return nullptr;
 1302|   141k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJRPKNS0_10IdentifierEDniDnEEEPT_DpOT0_:
  577|   272k|    {
  578|   272k|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|   272k|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 361, False: 272k]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|    361|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|    361|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|    361|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|    361|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 361]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|  58.1k|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 58.1k, False: 361]
  ------------------
  597|  58.1k|                heap.markFrom(sourceVal.second);
  598|  58.1k|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|    361|            heap.sweep();
  602|    361|        }
  603|   272k|        return r;
  604|   272k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter11builtinCharERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1335|    292|    {
 1336|    292|        validateBuiltinArgs(loc, "char", args, {Value::NUMBER});
 1337|    292|        long l = long(args[0].v.d);
 1338|    292|        if (l < 0) {
  ------------------
  |  Branch (1338:13): [True: 48, False: 244]
  ------------------
 1339|     48|            std::stringstream ss;
 1340|     48|            ss << "codepoints must be >= 0, got " << l;
 1341|     48|            throw makeError(loc, ss.str());
 1342|     48|        }
 1343|    244|        if (l >= JSONNET_CODEPOINT_MAX) {
  ------------------
  |  |   22|    244|#define JSONNET_CODEPOINT_MAX 0x110000
  ------------------
  |  Branch (1343:13): [True: 27, False: 217]
  ------------------
 1344|     27|            std::stringstream ss;
 1345|     27|            ss << "invalid unicode codepoint, got " << l;
 1346|     27|            throw makeError(loc, ss.str());
 1347|     27|        }
 1348|    217|        char32_t c = l;
 1349|    217|        scratch = makeString(UString(&c, 1));
 1350|    217|        return nullptr;
 1351|    244|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10builtinLogERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1354|  1.62k|    {
 1355|  1.62k|        validateBuiltinArgs(loc, "log", args, {Value::NUMBER});
 1356|  1.62k|        scratch = makeNumberCheck(loc, std::log(args[0].v.d));
 1357|  1.62k|        return nullptr;
 1358|  1.62k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter13builtinModuloERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1386|  72.9k|    {
 1387|  72.9k|        validateBuiltinArgs(loc, "modulo", args, {Value::NUMBER, Value::NUMBER});
 1388|  72.9k|        double a = args[0].v.d;
 1389|  72.9k|        double b = args[1].v.d;
 1390|  72.9k|        if (b == 0)
  ------------------
  |  Branch (1390:13): [True: 20, False: 72.8k]
  ------------------
 1391|     20|            throw makeError(loc, "division by zero.");
 1392|  72.8k|        scratch = makeNumberCheck(loc, std::fmod(a, b));
 1393|  72.8k|        return nullptr;
 1394|  72.9k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack3popEv:
  301|  80.6M|    {
  302|  80.6M|        if (top().isCall())
  ------------------
  |  Branch (302:13): [True: 31.4M, False: 49.2M]
  ------------------
  303|  31.4M|            calls--;
  304|  80.6M|        stack.pop_back();
  305|  80.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter22builtinPrimitiveEqualsERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1422|  3.52M|    {
 1423|  3.52M|        if (args.size() != 2) {
  ------------------
  |  Branch (1423:13): [True: 0, False: 3.52M]
  ------------------
 1424|      0|            std::stringstream ss;
 1425|      0|            ss << "primitiveEquals takes 2 parameters, got " << args.size();
 1426|      0|            throw makeError(loc, ss.str());
 1427|      0|        }
 1428|  3.52M|        if (args[0].t != args[1].t) {
  ------------------
  |  Branch (1428:13): [True: 0, False: 3.52M]
  ------------------
 1429|      0|            scratch = makeBoolean(false);
 1430|      0|            return nullptr;
 1431|      0|        }
 1432|  3.52M|        bool r;
 1433|  3.52M|        switch (args[0].t) {
 1434|  66.5k|            case Value::BOOLEAN: r = args[0].v.b == args[1].v.b; break;
  ------------------
  |  Branch (1434:13): [True: 66.5k, False: 3.46M]
  ------------------
 1435|       |
 1436|   146k|            case Value::NUMBER: r = args[0].v.d == args[1].v.d; break;
  ------------------
  |  Branch (1436:13): [True: 146k, False: 3.38M]
  ------------------
 1437|       |
 1438|  3.26M|            case Value::STRING:
  ------------------
  |  Branch (1438:13): [True: 3.26M, False: 266k]
  ------------------
 1439|  3.26M|                r = static_cast<HeapString *>(args[0].v.h)->value ==
 1440|  3.26M|                    static_cast<HeapString *>(args[1].v.h)->value;
 1441|  3.26M|                break;
 1442|       |
 1443|  53.6k|            case Value::NULL_TYPE: r = true; break;
  ------------------
  |  Branch (1443:13): [True: 53.6k, False: 3.47M]
  ------------------
 1444|       |
 1445|      0|            case Value::FUNCTION: throw makeError(loc, "cannot test equality of functions"); break;
  ------------------
  |  Branch (1445:13): [True: 0, False: 3.52M]
  ------------------
 1446|       |
 1447|      0|            default:
  ------------------
  |  Branch (1447:13): [True: 0, False: 3.52M]
  ------------------
 1448|      0|                throw makeError(loc,
 1449|      0|                                "primitiveEquals operates on primitive "
 1450|      0|                                "types, got " +
 1451|      0|                                    type_str(args[0]));
 1452|  3.52M|        }
 1453|  3.52M|        scratch = makeBoolean(r);
 1454|  3.52M|        return nullptr;
 1455|  3.52M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter18makeBuiltinFromASTEPKNS0_19BuiltinFunctionBodyE:
  658|  6.49M|    {
  659|  6.49M|        HeapClosure::Params hc_params;
  660|  10.2M|        for (const auto p : body->params) {
  ------------------
  |  Branch (660:27): [True: 10.2M, False: 6.49M]
  ------------------
  661|  10.2M|            hc_params.emplace_back(p, nullptr);
  662|  10.2M|        }
  663|  6.49M|        Value r;
  664|  6.49M|        r.t = Value::FUNCTION;
  665|  6.49M|        r.v.h = makeHeap<HeapClosure>(BindingFrame(), nullptr, 0, hc_params, body, body->name);
  666|  6.49M|        return r;
  667|  6.49M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_11HeapClosureEJNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEEDniRNS5_6vectorINS4_5ParamENSE_ISL_EEEERPKNS0_19BuiltinFunctionBodyERKNS5_12basic_stringIcNS5_11char_traitsIcEENSE_IcEEEEEEEPT_DpOT0_:
  577|  6.49M|    {
  578|  6.49M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  6.49M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 6.99k, False: 6.49M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  6.99k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  6.99k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  6.99k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  6.99k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 6.99k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|  1.12M|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 1.12M, False: 6.99k]
  ------------------
  597|  1.12M|                heap.markFrom(sourceVal.second);
  598|  1.12M|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  6.99k|            heap.sweep();
  602|  6.99k|        }
  603|  6.49M|        return r;
  604|  6.49M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter13builtinSubstrERKNS0_13LocationRangeERKNSt3__16vectorINS1_5ValueENS6_9allocatorIS8_EEEE:
 1599|    249|    {
 1600|    249|        validateBuiltinArgs(loc, "substr", args, {Value::STRING, Value::NUMBER, Value::NUMBER});
 1601|    249|        const auto *str = static_cast<const HeapString *>(args[0].v.h);
 1602|    249|        long from = long(args[1].v.d);
 1603|    249|        long len = long(args[2].v.d);
 1604|    249|        if (from < 0) {
  ------------------
  |  Branch (1604:13): [True: 0, False: 249]
  ------------------
 1605|      0|            std::stringstream ss;
 1606|      0|            ss << "substr second parameter should be greater than zero, got " << from;
 1607|      0|            throw makeError(loc, ss.str());
 1608|      0|        }
 1609|    249|        if (len < 0) {
  ------------------
  |  Branch (1609:13): [True: 0, False: 249]
  ------------------
 1610|      0|            std::stringstream ss;
 1611|      0|            ss << "substr third parameter should be greater than zero, got " << len;
 1612|      0|            throw makeError(loc, ss.str());
 1613|      0|        }
 1614|    249|        if (static_cast<unsigned long>(from) > str->value.size()) {
  ------------------
  |  Branch (1614:13): [True: 0, False: 249]
  ------------------
 1615|      0|            scratch = makeString(UString());
 1616|      0|            return nullptr;
 1617|      0|        }
 1618|    249|        if (size_t(len + from) > str->value.size()) {
  ------------------
  |  Branch (1618:13): [True: 0, False: 249]
  ------------------
 1619|      0|          len = str->value.size() - from;
 1620|      0|        }
 1621|    249|        scratch = makeString(str->value.substr(from, len));
 1622|    249|        return nullptr;
 1623|    249|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJDnDniPKNS0_3ASTEEEEPT_DpOT0_:
  577|  3.46k|    {
  578|  3.46k|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  3.46k|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 0, False: 3.46k]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|      0|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|      0|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|      0|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|      0|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 0]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|      0|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 0, False: 0]
  ------------------
  597|      0|                heap.markFrom(sourceVal.second);
  598|      0|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|      0|            heap.sweep();
  602|      0|        }
  603|  3.46k|        return r;
  604|  3.46k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack7newCallERKNS0_13LocationRangeEPNS1_10HeapEntityEPNS1_10HeapObjectEjRKNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENSA_4lessISE_EENSA_9allocatorINSA_4pairIKSE_SG_EEEEEE:
  418|  32.0M|    {
  419|  32.0M|        tailCallTrimStack();
  420|  32.0M|        if (calls >= limit) {
  ------------------
  |  Branch (420:13): [True: 549, False: 32.0M]
  ------------------
  421|    549|            throw makeError(loc, "max stack frames exceeded.");
  422|    549|        }
  423|  32.0M|        stack.emplace_back(FRAME_CALL, loc);
  424|  32.0M|        calls++;
  425|  32.0M|        top().context = context;
  426|  32.0M|        top().self = self;
  427|  32.0M|        top().offset = offset;
  428|  32.0M|        top().bindings = up_values;
  429|  32.0M|        top().tailCall = false;
  430|       |
  431|  32.0M|#ifndef NDEBUG
  432|  34.9M|        for (const auto &bind : up_values) {
  ------------------
  |  Branch (432:31): [True: 34.9M, False: 32.0M]
  ------------------
  433|  34.9M|            if (bind.second == nullptr) {
  ------------------
  |  Branch (433:17): [True: 0, False: 34.9M]
  ------------------
  434|      0|                std::cerr << "INTERNAL ERROR: No binding for variable "
  435|      0|                          << encode_utf8(bind.first->name) << std::endl;
  436|      0|                std::abort();
  437|      0|            }
  438|  34.9M|        }
  439|  32.0M|#endif
  440|  32.0M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack17tailCallTrimStackEv:
  394|  32.0M|    {
  395|  32.7M|        for (int i = stack.size() - 1; i >= 0; --i) {
  ------------------
  |  Branch (395:40): [True: 32.2M, False: 477k]
  ------------------
  396|  32.2M|            switch (stack[i].kind) {
  397|  8.75M|                case FRAME_CALL: {
  ------------------
  |  Branch (397:17): [True: 8.75M, False: 23.4M]
  ------------------
  398|  8.75M|                    if (!stack[i].tailCall || stack[i].thunks.size() > 0) {
  ------------------
  |  Branch (398:25): [True: 5.77M, False: 2.98M]
  |  Branch (398:47): [True: 2.69M, False: 295k]
  ------------------
  399|  8.46M|                        return;
  400|  8.46M|                    }
  401|       |                    // Remove all stack frames including this one.
  402|  1.14M|                    while (stack.size() > unsigned(i))
  ------------------
  |  Branch (402:28): [True: 846k, False: 295k]
  ------------------
  403|   846k|                        stack.pop_back();
  404|   295k|                    calls--;
  405|   295k|                    return;
  406|  8.75M|                } break;
  407|       |
  408|   722k|                case FRAME_LOCAL: break;
  ------------------
  |  Branch (408:17): [True: 722k, False: 31.5M]
  ------------------
  409|       |
  410|  22.7M|                default: return;
  ------------------
  |  Branch (410:17): [True: 22.7M, False: 9.48M]
  ------------------
  411|  32.2M|            }
  412|  32.2M|        }
  413|  32.0M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15FrameC2ERKNS1_9FrameKindERKNS0_13LocationRangeE:
  209|  40.9M|        : kind(kind),
  210|  40.9M|          ast(nullptr),
  211|  40.9M|          location(location),
  212|  40.9M|          tailCall(false),
  213|  40.9M|          elementId(0),
  214|  40.9M|          first(false),
  215|  40.9M|          context(NULL),
  216|  40.9M|          self(NULL),
  217|  40.9M|          offset(0)
  218|  40.9M|    {
  219|  40.9M|        val.t = Value::NULL_TYPE;
  220|  40.9M|        val2.t = Value::NULL_TYPE;
  221|  40.9M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter22prepareSourceValThunksEv:
  887|  3.46k|    void prepareSourceValThunks() {
  888|   557k|        for (const auto &field : stdlibAST->fields) {
  ------------------
  |  Branch (888:32): [True: 557k, False: 3.46k]
  ------------------
  889|   557k|            AST *nameAST = field.name;
  890|   557k|            if (nameAST->type != AST_LITERAL_STRING) {
  ------------------
  |  Branch (890:17): [True: 0, False: 557k]
  ------------------
  891|       |                // Skip any fields without a known name.
  892|      0|                continue;
  893|      0|            }
  894|   557k|            UString name = dynamic_cast<LiteralString *>(nameAST)->value;
  895|       |
  896|   557k|            sourceFuncIds.emplace_back(new Identifier(name));
  897|   557k|            auto *th = makeHeap<HeapThunk>(sourceFuncIds.back().get(), stdObject, 0, field.body);
  898|   557k|            sourceVals[encode_utf8(name)] = th;
  899|   557k|        }
  900|  3.46k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJPNS0_10IdentifierERPNS1_10HeapObjectEiRKPNS0_3ASTEEEEPT_DpOT0_:
  577|   557k|    {
  578|   557k|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|   557k|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 0, False: 557k]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|      0|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|      0|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|      0|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|      0|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 0]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|      0|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 0, False: 0]
  ------------------
  597|      0|                heap.markFrom(sourceVal.second);
  598|      0|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|      0|            heap.sweep();
  602|      0|        }
  603|   557k|        return r;
  604|   557k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15StackD2Ev:
  260|  3.46k|    ~Stack(void) {}
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8evaluateEPKNS0_3ASTEj:
 2096|  2.37M|    {
 2097|  94.4M|    recurse:
 2098|       |
 2099|  94.4M|        switch (ast_->type) {
 2100|  8.39M|            case AST_APPLY: {
  ------------------
  |  Branch (2100:13): [True: 8.39M, False: 86.0M]
  ------------------
 2101|  8.39M|                const auto &ast = *static_cast<const Apply *>(ast_);
 2102|  8.39M|                stack.newFrame(FRAME_APPLY_TARGET, ast_);
 2103|  8.39M|                ast_ = ast.target;
 2104|  8.39M|                goto recurse;
 2105|      0|            } break;
 2106|       |
 2107|   591k|            case AST_ARRAY: {
  ------------------
  |  Branch (2107:13): [True: 591k, False: 93.8M]
  ------------------
 2108|   591k|                const auto &ast = *static_cast<const Array *>(ast_);
 2109|   591k|                HeapObject *self;
 2110|   591k|                unsigned offset;
 2111|   591k|                stack.getSelfBinding(self, offset);
 2112|   591k|                scratch = makeArray({});
 2113|   591k|                auto &elements = static_cast<HeapArray *>(scratch.v.h)->elements;
 2114|  2.37M|                for (const auto &el : ast.elements) {
  ------------------
  |  Branch (2114:37): [True: 2.37M, False: 591k]
  ------------------
 2115|  2.37M|                    auto *el_th = makeHeap<HeapThunk>(idArrayElement, self, offset, el.expr);
 2116|  2.37M|                    el_th->upValues = capture(el.expr->freeVariables);
 2117|  2.37M|                    elements.push_back(el_th);
 2118|  2.37M|                }
 2119|   591k|            } break;
 2120|       |
 2121|  3.41M|            case AST_BINARY: {
  ------------------
  |  Branch (2121:13): [True: 3.41M, False: 91.0M]
  ------------------
 2122|  3.41M|                const auto &ast = *static_cast<const Binary *>(ast_);
 2123|  3.41M|                stack.newFrame(FRAME_BINARY_LEFT, ast_);
 2124|  3.41M|                ast_ = ast.left;
 2125|  3.41M|                goto recurse;
 2126|      0|            } break;
 2127|       |
 2128|  6.49M|            case AST_BUILTIN_FUNCTION: {
  ------------------
  |  Branch (2128:13): [True: 6.49M, False: 87.9M]
  ------------------
 2129|  6.49M|                const auto &ast = *static_cast<const BuiltinFunction *>(ast_);
 2130|  6.49M|                scratch = makeBuiltinFromAST(ast.body);
 2131|  6.49M|            } break;
 2132|       |
 2133|      0|            case AST_BUILTIN_FUNCTION_BODY: {
  ------------------
  |  Branch (2133:13): [True: 0, False: 94.4M]
  ------------------
 2134|       |                // We can reach here if we end up evaluating a HeapThunk that was constructed
 2135|       |                // from a built-in function. Synthesize a FRAME_BUILTIN_FORCE_THUNKS to call it.
 2136|      0|                assert(stack.top().isCall());
  ------------------
  |  Branch (2136:17): [True: 0, False: 0]
  ------------------
 2137|      0|                const auto fbody = static_cast<const BuiltinFunctionBody*>(ast_);
 2138|       |
 2139|      0|                const BindingFrame up_values = stack.top().bindings;
 2140|      0|                std::vector<HeapThunk*> arg_thunks;
 2141|      0|                for (const auto &p : fbody->params) {
  ------------------
  |  Branch (2141:36): [True: 0, False: 0]
  ------------------
 2142|      0|                    const auto it = up_values.find(p);
 2143|      0|                    if (it != up_values.end()) {
  ------------------
  |  Branch (2143:25): [True: 0, False: 0]
  ------------------
 2144|      0|                        arg_thunks.push_back(it->second);
 2145|      0|                    } else {
 2146|      0|                        JSONNET_UNREACHABLE();
  ------------------
  |  |   53|      0|  do {                                                             \
  |  |   54|      0|    std::cerr << __FILE__ << ":" << __LINE__                       \
  |  |   55|      0|              << ": INTERNAL ERROR: reached unreachable code path" \
  |  |   56|      0|              << std::endl;                                        \
  |  |   57|      0|    abort();                                                       \
  |  |   58|      0|  } while (0)
  |  |  ------------------
  |  |  |  Branch (58:12): [Folded, False: 0]
  |  |  ------------------
  ------------------
 2147|      0|                    }
 2148|      0|                }
 2149|      0|                const LocationRange &loc = stack.top().location;
 2150|       |
 2151|      0|                stack.newFrame(FRAME_BUILTIN_FORCE_THUNKS, loc);
 2152|      0|                stack.top().bindings = up_values;
 2153|      0|                stack.top().thunks = arg_thunks;
 2154|      0|                stack.top().val = makeBuiltinFromAST(fbody);
 2155|      0|            } break;
 2156|       |
 2157|  5.30M|            case AST_CONDITIONAL: {
  ------------------
  |  Branch (2157:13): [True: 5.30M, False: 89.1M]
  ------------------
 2158|  5.30M|                const auto &ast = *static_cast<const Conditional *>(ast_);
 2159|  5.30M|                stack.newFrame(FRAME_IF, ast_);
 2160|  5.30M|                ast_ = ast.cond;
 2161|  5.30M|                goto recurse;
 2162|      0|            } break;
 2163|       |
 2164|  3.32k|            case AST_ERROR: {
  ------------------
  |  Branch (2164:13): [True: 3.32k, False: 94.4M]
  ------------------
 2165|  3.32k|                const auto &ast = *static_cast<const Error *>(ast_);
 2166|  3.32k|                stack.newFrame(FRAME_ERROR, ast_);
 2167|  3.32k|                ast_ = ast.expr;
 2168|  3.32k|                goto recurse;
 2169|      0|            } break;
 2170|       |
 2171|  1.60M|            case AST_FUNCTION: {
  ------------------
  |  Branch (2171:13): [True: 1.60M, False: 92.8M]
  ------------------
 2172|  1.60M|                const auto &ast = *static_cast<const Function *>(ast_);
 2173|  1.60M|                auto env = capture(ast.freeVariables);
 2174|  1.60M|                HeapObject *self;
 2175|  1.60M|                unsigned offset;
 2176|  1.60M|                stack.getSelfBinding(self, offset);
 2177|  1.60M|                HeapClosure::Params params;
 2178|  1.60M|                params.reserve(ast.params.size());
 2179|  3.01M|                for (const auto &p : ast.params) {
  ------------------
  |  Branch (2179:36): [True: 3.01M, False: 1.60M]
  ------------------
 2180|  3.01M|                    params.emplace_back(p.id, p.expr);
 2181|  3.01M|                }
 2182|  1.60M|                scratch = makeClosure(env, self, offset, params, ast.body);
 2183|  1.60M|            } break;
 2184|       |
 2185|     23|            case AST_IMPORT: {
  ------------------
  |  Branch (2185:13): [True: 23, False: 94.4M]
  ------------------
 2186|     23|                const auto &ast = *static_cast<const Import *>(ast_);
 2187|     23|                HeapThunk *thunk = import(ast.location, ast.file);
 2188|     23|                if (thunk->filled) {
  ------------------
  |  Branch (2188:21): [True: 0, False: 23]
  ------------------
 2189|      0|                    scratch = thunk->content;
 2190|     23|                } else {
 2191|     23|                    stack.newCall(ast.location, thunk, thunk->self, thunk->offset, thunk->upValues);
 2192|     23|                    ast_ = thunk->body;
 2193|     23|                    goto recurse;
 2194|     23|                }
 2195|     23|            } break;
 2196|       |
 2197|      5|            case AST_IMPORTSTR: {
  ------------------
  |  Branch (2197:13): [True: 5, False: 94.4M]
  ------------------
 2198|      5|                const auto &ast = *static_cast<const Importstr *>(ast_);
 2199|      5|                const ImportCacheValue *value = importData(ast.location, ast.file);
 2200|      5|                scratch = makeString(decode_utf8(value->content));
 2201|      5|            } break;
 2202|       |
 2203|      6|            case AST_IMPORTBIN: {
  ------------------
  |  Branch (2203:13): [True: 6, False: 94.4M]
  ------------------
 2204|      6|                const auto &ast = *static_cast<const Importbin *>(ast_);
 2205|      6|                const ImportCacheValue *value = importData(ast.location, ast.file);
 2206|      6|                scratch = makeArray({});
 2207|      6|                auto &elements = static_cast<HeapArray *>(scratch.v.h)->elements;
 2208|      6|                elements.reserve(value->content.size());
 2209|      6|                for (const auto c : value->content) {
  ------------------
  |  Branch (2209:35): [True: 0, False: 6]
  ------------------
 2210|      0|                    auto *th = makeHeap<HeapThunk>(idArrayElement, nullptr, 0, nullptr);
 2211|      0|                    elements.push_back(th);
 2212|      0|                    th->fill(makeNumber(uint8_t(c)));
 2213|      0|                }
 2214|      6|            } break;
 2215|       |
 2216|  1.32M|            case AST_IN_SUPER: {
  ------------------
  |  Branch (2216:13): [True: 1.32M, False: 93.1M]
  ------------------
 2217|  1.32M|                const auto &ast = *static_cast<const InSuper *>(ast_);
 2218|  1.32M|                stack.newFrame(FRAME_IN_SUPER_ELEMENT, ast_);
 2219|  1.32M|                ast_ = ast.element;
 2220|  1.32M|                goto recurse;
 2221|     23|            } break;
 2222|       |
 2223|  8.70M|            case AST_INDEX: {
  ------------------
  |  Branch (2223:13): [True: 8.70M, False: 85.7M]
  ------------------
 2224|  8.70M|                const auto &ast = *static_cast<const Index *>(ast_);
 2225|  8.70M|                stack.newFrame(FRAME_INDEX_TARGET, ast_);
 2226|  8.70M|                ast_ = ast.target;
 2227|  8.70M|                goto recurse;
 2228|     23|            } break;
 2229|       |
 2230|  4.34M|            case AST_LOCAL: {
  ------------------
  |  Branch (2230:13): [True: 4.34M, False: 90.1M]
  ------------------
 2231|  4.34M|                const auto &ast = *static_cast<const Local *>(ast_);
 2232|  4.34M|                stack.newFrame(FRAME_LOCAL, ast_);
 2233|  4.34M|                Frame &f = stack.top();
 2234|       |                // First build all the thunks and bind them.
 2235|  4.34M|                HeapObject *self;
 2236|  4.34M|                unsigned offset;
 2237|  4.34M|                stack.getSelfBinding(self, offset);
 2238|  14.6M|                for (const auto &bind : ast.binds) {
  ------------------
  |  Branch (2238:39): [True: 14.6M, False: 4.34M]
  ------------------
 2239|       |                    // Note that these 2 lines must remain separate to avoid the GC running
 2240|       |                    // when bindings has a nullptr for key bind.first.
 2241|  14.6M|                    auto *th = makeHeap<HeapThunk>(bind.var, self, offset, bind.body);
 2242|  14.6M|                    f.bindings[bind.var] = th;
 2243|  14.6M|                }
 2244|       |                // Now capture the environment (including the new thunks, to make cycles).
 2245|  14.6M|                for (const auto &bind : ast.binds) {
  ------------------
  |  Branch (2245:39): [True: 14.6M, False: 4.34M]
  ------------------
 2246|  14.6M|                    auto *thunk = f.bindings[bind.var];
 2247|  14.6M|                    thunk->upValues = capture(bind.body->freeVariables);
 2248|  14.6M|                }
 2249|  4.34M|                ast_ = ast.body;
 2250|  4.34M|                goto recurse;
 2251|     23|            } break;
 2252|       |
 2253|   449k|            case AST_LITERAL_BOOLEAN: {
  ------------------
  |  Branch (2253:13): [True: 449k, False: 94.0M]
  ------------------
 2254|   449k|                const auto &ast = *static_cast<const LiteralBoolean *>(ast_);
 2255|   449k|                scratch = makeBoolean(ast.value);
 2256|   449k|            } break;
 2257|       |
 2258|  2.37M|            case AST_LITERAL_NUMBER: {
  ------------------
  |  Branch (2258:13): [True: 2.37M, False: 92.0M]
  ------------------
 2259|  2.37M|                const auto &ast = *static_cast<const LiteralNumber *>(ast_);
 2260|  2.37M|                scratch = makeNumberCheck(ast_->location, ast.value);
 2261|  2.37M|            } break;
 2262|       |
 2263|  24.0M|            case AST_LITERAL_STRING: {
  ------------------
  |  Branch (2263:13): [True: 24.0M, False: 70.4M]
  ------------------
 2264|  24.0M|                const auto &ast = *static_cast<const LiteralString *>(ast_);
 2265|  24.0M|                scratch = makeString(ast.value);
 2266|  24.0M|            } break;
 2267|       |
 2268|   105k|            case AST_LITERAL_NULL: {
  ------------------
  |  Branch (2268:13): [True: 105k, False: 94.3M]
  ------------------
 2269|   105k|                scratch = makeNull();
 2270|   105k|            } break;
 2271|       |
 2272|  1.43M|            case AST_DESUGARED_OBJECT: {
  ------------------
  |  Branch (2272:13): [True: 1.43M, False: 93.0M]
  ------------------
 2273|  1.43M|                const auto &ast = *static_cast<const DesugaredObject *>(ast_);
 2274|  1.43M|                if (ast.fields.empty()) {
  ------------------
  |  Branch (2274:21): [True: 938k, False: 498k]
  ------------------
 2275|   938k|                    auto env = capture(ast.freeVariables);
 2276|   938k|                    std::map<const Identifier *, HeapSimpleObject::Field> fields;
 2277|   938k|                    scratch = makeObject<HeapSimpleObject>(env, fields, ast.asserts);
 2278|   938k|                } else {
 2279|   498k|                    auto env = capture(ast.freeVariables);
 2280|   498k|                    stack.newFrame(FRAME_OBJECT, ast_);
 2281|   498k|                    auto fit = ast.fields.begin();
 2282|   498k|                    stack.top().fit = fit;
 2283|   498k|                    ast_ = fit->name;
 2284|   498k|                    goto recurse;
 2285|   498k|                }
 2286|  1.43M|            } break;
 2287|       |
 2288|   938k|            case AST_OBJECT_COMPREHENSION_SIMPLE: {
  ------------------
  |  Branch (2288:13): [True: 95, False: 94.4M]
  ------------------
 2289|     95|                const auto &ast = *static_cast<const ObjectComprehensionSimple *>(ast_);
 2290|     95|                stack.newFrame(FRAME_OBJECT_COMP_ARRAY, ast_);
 2291|     95|                ast_ = ast.array;
 2292|     95|                goto recurse;
 2293|  1.43M|            } break;
 2294|       |
 2295|  1.62M|            case AST_SELF: {
  ------------------
  |  Branch (2295:13): [True: 1.62M, False: 92.8M]
  ------------------
 2296|  1.62M|                scratch.t = Value::OBJECT;
 2297|  1.62M|                HeapObject *self;
 2298|  1.62M|                unsigned offset;
 2299|  1.62M|                stack.getSelfBinding(self, offset);
 2300|  1.62M|                scratch.v.h = self;
 2301|  1.62M|            } break;
 2302|       |
 2303|  1.16M|            case AST_SUPER_INDEX: {
  ------------------
  |  Branch (2303:13): [True: 1.16M, False: 93.2M]
  ------------------
 2304|  1.16M|                const auto &ast = *static_cast<const SuperIndex *>(ast_);
 2305|  1.16M|                stack.newFrame(FRAME_SUPER_INDEX, ast_);
 2306|  1.16M|                ast_ = ast.index;
 2307|  1.16M|                goto recurse;
 2308|  1.43M|            } break;
 2309|       |
 2310|  2.41M|            case AST_UNARY: {
  ------------------
  |  Branch (2310:13): [True: 2.41M, False: 92.0M]
  ------------------
 2311|  2.41M|                const auto &ast = *static_cast<const Unary *>(ast_);
 2312|  2.41M|                stack.newFrame(FRAME_UNARY, ast_);
 2313|  2.41M|                ast_ = ast.expr;
 2314|  2.41M|                goto recurse;
 2315|  1.43M|            } break;
 2316|       |
 2317|  20.6M|            case AST_VAR: {
  ------------------
  |  Branch (2317:13): [True: 20.6M, False: 73.7M]
  ------------------
 2318|  20.6M|                const auto &ast = *static_cast<const Var *>(ast_);
 2319|  20.6M|                auto *thunk = stack.lookUpVar(ast.id);
 2320|  20.6M|                if (thunk == nullptr) {
  ------------------
  |  Branch (2320:21): [True: 0, False: 20.6M]
  ------------------
 2321|      0|                    std::cerr << "INTERNAL ERROR: Could not bind variable: "
 2322|      0|                              << encode_utf8(ast.id->name) << " at "
 2323|      0|                              << ast.location << std::endl;
 2324|      0|                    std::abort();
 2325|      0|                }
 2326|  20.6M|                if (thunk->filled) {
  ------------------
  |  Branch (2326:21): [True: 15.5M, False: 5.14M]
  ------------------
 2327|  15.5M|                    scratch = thunk->content;
 2328|  15.5M|                } else {
 2329|  5.14M|                    stack.newCall(ast.location, thunk, thunk->self, thunk->offset, thunk->upValues);
 2330|  5.14M|                    ast_ = thunk->body;
 2331|  5.14M|                    goto recurse;
 2332|  5.14M|                }
 2333|  20.6M|            } break;
 2334|       |
 2335|  15.5M|            default:
  ------------------
  |  Branch (2335:13): [True: 0, False: 94.4M]
  ------------------
 2336|      0|                std::cerr << "INTERNAL ERROR: Unknown AST: " << ast_->type << std::endl;
 2337|      0|                std::abort();
 2338|  94.4M|        }
 2339|       |
 2340|       |        // To evaluate another AST, set ast to it, then goto recurse.
 2341|       |        // To pop, exit the switch or goto popframe
 2342|       |        // To change the frame and re-enter the switch, goto replaceframe
 2343|   108M|        while (stack.size() > initial_stack_size) {
  ------------------
  |  Branch (2343:16): [True: 106M, False: 2.35M]
  ------------------
 2344|   106M|            Frame &f = stack.top();
 2345|   106M|            switch (f.kind) {
 2346|  8.39M|                case FRAME_APPLY_TARGET: {
  ------------------
  |  Branch (2346:17): [True: 8.39M, False: 97.9M]
  ------------------
 2347|  8.39M|                    const auto &ast = *static_cast<const Apply *>(f.ast);
 2348|  8.39M|                    if (scratch.t != Value::FUNCTION) {
  ------------------
  |  Branch (2348:25): [True: 12, False: 8.39M]
  ------------------
 2349|     12|                        throw makeError(ast.location,
 2350|     12|                                        "only functions can be called, got " + type_str(scratch));
 2351|     12|                    }
 2352|  8.39M|                    auto *func = static_cast<HeapClosure *>(scratch.v.h);
 2353|       |
 2354|  8.39M|                    std::set<const Identifier *> params_needed;
 2355|  14.0M|                    for (const auto &param : func->params) {
  ------------------
  |  Branch (2355:44): [True: 14.0M, False: 8.39M]
  ------------------
 2356|  14.0M|                        params_needed.insert(param.id);
 2357|  14.0M|                    }
 2358|       |
 2359|       |                    // Create thunks for arguments.
 2360|  8.39M|                    std::vector<HeapThunk *> positional_args;
 2361|  8.39M|                    BindingFrame args;
 2362|  8.39M|                    bool got_named = false;
 2363|  22.4M|                    for (std::size_t i = 0; i < ast.args.size(); ++i) {
  ------------------
  |  Branch (2363:45): [True: 14.0M, False: 8.39M]
  ------------------
 2364|  14.0M|                        const auto &arg = ast.args[i];
 2365|       |
 2366|  14.0M|                        const Identifier *name;
 2367|  14.0M|                        if (arg.id != nullptr) {
  ------------------
  |  Branch (2367:29): [True: 0, False: 14.0M]
  ------------------
 2368|      0|                            got_named = true;
 2369|      0|                            name = arg.id;
 2370|  14.0M|                        } else {
 2371|  14.0M|                            if (got_named) {
  ------------------
  |  Branch (2371:33): [True: 0, False: 14.0M]
  ------------------
 2372|      0|                                std::stringstream ss;
 2373|      0|                                ss << "internal error: got positional param after named at index "
 2374|      0|                                   << i;
 2375|      0|                                throw makeError(ast.location, ss.str());
 2376|      0|                            }
 2377|  14.0M|                            if (i >= func->params.size()) {
  ------------------
  |  Branch (2377:33): [True: 0, False: 14.0M]
  ------------------
 2378|      0|                                std::stringstream ss;
 2379|      0|                                ss << "too many args, function has " << func->params.size()
 2380|      0|                                   << " parameter(s)";
 2381|      0|                                throw makeError(ast.location, ss.str());
 2382|      0|                            }
 2383|  14.0M|                            name = func->params[i].id;
 2384|  14.0M|                        }
 2385|       |                        // Special case for builtin functions -- leave identifier blank for
 2386|       |                        // them in the thunk.  This removes the thunk frame from the stacktrace.
 2387|  14.0M|                        const Identifier *name_ = func->isBuiltin() ? nullptr : name;
  ------------------
  |  Branch (2387:51): [True: 10.2M, False: 3.84M]
  ------------------
 2388|  14.0M|                        HeapObject *self;
 2389|  14.0M|                        unsigned offset;
 2390|  14.0M|                        stack.getSelfBinding(self, offset);
 2391|  14.0M|                        auto *thunk = makeHeap<HeapThunk>(name_, self, offset, arg.expr);
 2392|  14.0M|                        thunk->upValues = capture(arg.expr->freeVariables);
 2393|       |                        // While making the thunks, keep them in a frame to avoid premature garbage
 2394|       |                        // collection.
 2395|  14.0M|                        f.thunks.push_back(thunk);
 2396|  14.0M|                        if (args.find(name) != args.end()) {
  ------------------
  |  Branch (2396:29): [True: 0, False: 14.0M]
  ------------------
 2397|      0|                            std::stringstream ss;
 2398|      0|                            ss << "binding parameter a second time: " << encode_utf8(name->name);
 2399|      0|                            throw makeError(ast.location, ss.str());
 2400|      0|                        }
 2401|  14.0M|                        args[name] = thunk;
 2402|  14.0M|                        if (params_needed.find(name) == params_needed.end()) {
  ------------------
  |  Branch (2402:29): [True: 0, False: 14.0M]
  ------------------
 2403|      0|                            std::stringstream ss;
 2404|      0|                            ss << "function has no parameter " << encode_utf8(name->name);
 2405|      0|                            throw makeError(ast.location, ss.str());
 2406|      0|                        }
 2407|  14.0M|                    }
 2408|       |
 2409|       |                    // For any func params for which there was no arg, create a thunk for those and
 2410|       |                    // bind the default argument.  Allow default thunks to see other params.  If no
 2411|       |                    // default argument than raise an error.
 2412|       |
 2413|       |                    // Raise errors for unbound params, create thunks (but don't fill in upvalues).
 2414|       |                    // This is a subset of f.thunks, so will not get garbage collected.
 2415|  8.39M|                    std::vector<HeapThunk *> def_arg_thunks;
 2416|  14.0M|                    for (const auto &param : func->params) {
  ------------------
  |  Branch (2416:44): [True: 14.0M, False: 8.39M]
  ------------------
 2417|  14.0M|                        if (args.find(param.id) != args.end())
  ------------------
  |  Branch (2417:29): [True: 14.0M, False: 38]
  ------------------
 2418|  14.0M|                            continue;
 2419|     38|                        if (param.def == nullptr) {
  ------------------
  |  Branch (2419:29): [True: 9, False: 29]
  ------------------
 2420|      9|                            std::stringstream ss;
 2421|      9|                            ss << "function parameter " << encode_utf8(param.id->name)
 2422|      9|                               << " not bound in call.";
 2423|      9|                            throw makeError(ast.location, ss.str());
 2424|      9|                        }
 2425|       |
 2426|       |                        // Special case for builtin functions -- leave identifier blank for
 2427|       |                        // them in the thunk.  This removes the thunk frame from the stacktrace.
 2428|     29|                        const Identifier *name_ = func->isBuiltin() ? nullptr : param.id;
  ------------------
  |  Branch (2428:51): [True: 0, False: 29]
  ------------------
 2429|     29|                        auto *thunk =
 2430|     29|                            makeHeap<HeapThunk>(name_, func->self, func->offset, param.def);
 2431|     29|                        f.thunks.push_back(thunk);
 2432|     29|                        def_arg_thunks.push_back(thunk);
 2433|     29|                        args[param.id] = thunk;
 2434|     29|                    }
 2435|       |
 2436|  8.39M|                    BindingFrame up_values = func->upValues;
 2437|  8.39M|                    up_values.insert(args.begin(), args.end());
 2438|       |
 2439|       |                    // Fill in upvalues
 2440|  8.39M|                    for (HeapThunk *thunk : def_arg_thunks) {
  ------------------
  |  Branch (2440:43): [True: 21, False: 8.39M]
  ------------------
 2441|     21|                        thunk->upValues = up_values;
 2442|     21|                    }
 2443|       |
 2444|       |                    // Cache these, because pop will invalidate them.
 2445|  8.39M|                    std::vector<HeapThunk *> thunks_copy = f.thunks;
 2446|       |
 2447|  8.39M|                    const AST *f_ast = f.ast;
 2448|  8.39M|                    stack.pop();
 2449|       |
 2450|  8.39M|                    if (func->isBuiltin()) {
  ------------------
  |  Branch (2450:25): [True: 6.49M, False: 1.89M]
  ------------------
 2451|       |                        // Built-in function.
 2452|       |                        // Give nullptr for self because no one looking at this frame will
 2453|       |                        // attempt to bind to self (it's native code).
 2454|  6.49M|                        stack.newFrame(FRAME_BUILTIN_FORCE_THUNKS, f_ast);
 2455|  6.49M|                        assert(func->upValues.empty());
  ------------------
  |  Branch (2455:25): [True: 6.49M, False: 0]
  ------------------
 2456|  6.49M|                        stack.top().bindings = up_values;
 2457|  6.49M|                        stack.top().thunks = thunks_copy;
 2458|  6.49M|                        stack.top().val = scratch;
 2459|  6.49M|                        goto replaceframe;
 2460|  6.49M|                    } else {
 2461|       |                        // User defined function.
 2462|  1.89M|                        stack.newCall(ast.location, func, func->self, func->offset, up_values);
 2463|  1.89M|                        if (ast.tailstrict) {
  ------------------
  |  Branch (2463:29): [True: 1.22M, False: 669k]
  ------------------
 2464|  1.22M|                            stack.top().tailCall = true;
 2465|  1.22M|                            if (thunks_copy.size() == 0) {
  ------------------
  |  Branch (2465:33): [True: 0, False: 1.22M]
  ------------------
 2466|       |                                // No need to force thunks, proceed straight to body.
 2467|      0|                                ast_ = func->body;
 2468|      0|                                goto recurse;
 2469|  1.22M|                            } else {
 2470|       |                                // The check for args.size() > 0
 2471|  1.22M|                                stack.top().thunks = thunks_copy;
 2472|  1.22M|                                stack.top().val = scratch;
 2473|  1.22M|                                goto replaceframe;
 2474|  1.22M|                            }
 2475|  1.22M|                        } else {
 2476|   669k|                            ast_ = func->body;
 2477|   669k|                            goto recurse;
 2478|   669k|                        }
 2479|  1.89M|                    }
 2480|  8.39M|                } break;
 2481|       |
 2482|  3.28M|                case FRAME_BINARY_LEFT: {
  ------------------
  |  Branch (2482:17): [True: 3.28M, False: 103M]
  ------------------
 2483|  3.28M|                    const auto &ast = *static_cast<const Binary *>(f.ast);
 2484|  3.28M|                    const Value &lhs = scratch;
 2485|  3.28M|                    if (lhs.t == Value::BOOLEAN) {
  ------------------
  |  Branch (2485:25): [True: 169k, False: 3.11M]
  ------------------
 2486|       |                        // Handle short-cut semantics
 2487|   169k|                        switch (ast.op) {
 2488|  80.7k|                            case BOP_AND: {
  ------------------
  |  Branch (2488:29): [True: 80.7k, False: 88.9k]
  ------------------
 2489|  80.7k|                                if (!lhs.v.b) {
  ------------------
  |  Branch (2489:37): [True: 4.50k, False: 76.2k]
  ------------------
 2490|  4.50k|                                    scratch = makeBoolean(false);
 2491|  4.50k|                                    goto popframe;
 2492|  4.50k|                                }
 2493|  80.7k|                            } break;
 2494|       |
 2495|  88.3k|                            case BOP_OR: {
  ------------------
  |  Branch (2495:29): [True: 88.3k, False: 81.4k]
  ------------------
 2496|  88.3k|                                if (lhs.v.b) {
  ------------------
  |  Branch (2496:37): [True: 1.35k, False: 86.9k]
  ------------------
 2497|  1.35k|                                    scratch = makeBoolean(true);
 2498|  1.35k|                                    goto popframe;
 2499|  1.35k|                                }
 2500|  88.3k|                            } break;
 2501|       |
 2502|  86.9k|                            default:;
  ------------------
  |  Branch (2502:29): [True: 612, False: 169k]
  ------------------
 2503|   169k|                        }
 2504|   169k|                    }
 2505|  3.27M|                    stack.top().kind = FRAME_BINARY_RIGHT;
 2506|  3.27M|                    stack.top().val = lhs;
 2507|  3.27M|                    ast_ = ast.right;
 2508|  3.27M|                    goto recurse;
 2509|  3.28M|                } break;
 2510|       |
 2511|  3.27M|                case FRAME_BINARY_RIGHT: {
  ------------------
  |  Branch (2511:17): [True: 3.27M, False: 103M]
  ------------------
 2512|  3.27M|                    stack.top().val2 = scratch;
 2513|  3.27M|                    stack.top().kind = FRAME_BINARY_OP;
 2514|  3.27M|                }
 2515|  3.27M|                [[fallthrough]];
 2516|  3.27M|                case FRAME_BINARY_OP: {
  ------------------
  |  Branch (2516:17): [True: 113, False: 106M]
  ------------------
 2517|  3.27M|                    const auto &ast = *static_cast<const Binary *>(f.ast);
 2518|  3.27M|                    const Value &lhs = stack.top().val;
 2519|  3.27M|                    const Value &rhs = stack.top().val2;
 2520|       |
 2521|       |                    // Handle cases where the LHS and RHS are not the same type.
 2522|  3.27M|                    if (lhs.t == Value::STRING || rhs.t == Value::STRING) {
  ------------------
  |  Branch (2522:25): [True: 831k, False: 2.44M]
  |  Branch (2522:51): [True: 168k, False: 2.27M]
  ------------------
 2523|   999k|                        if (ast.op == BOP_PLUS) {
  ------------------
  |  Branch (2523:29): [True: 908k, False: 91.0k]
  ------------------
 2524|       |                            // Handle co-ercions for string processing.
 2525|   908k|                            stack.top().kind = FRAME_STRING_CONCAT;
 2526|   908k|                            stack.top().val2 = rhs;
 2527|   908k|                            goto replaceframe;
 2528|   908k|                        }
 2529|   999k|                    }
 2530|  2.36M|                    switch (ast.op) {
 2531|       |                        // Equality can be used when the types don't match.
 2532|      0|                        case BOP_MANIFEST_EQUAL:
  ------------------
  |  Branch (2532:25): [True: 0, False: 2.36M]
  ------------------
 2533|      0|                            std::cerr << "INTERNAL ERROR: Equals not desugared" << std::endl;
 2534|      0|                            abort();
 2535|       |
 2536|       |                        // Equality can be used when the types don't match.
 2537|      0|                        case BOP_MANIFEST_UNEQUAL:
  ------------------
  |  Branch (2537:25): [True: 0, False: 2.36M]
  ------------------
 2538|      0|                            std::cerr << "INTERNAL ERROR: Notequals not desugared" << std::endl;
 2539|      0|                            abort();
 2540|       |
 2541|       |                        // e in e
 2542|    172|                        case BOP_IN: {
  ------------------
  |  Branch (2542:25): [True: 172, False: 2.36M]
  ------------------
 2543|    172|                            if (lhs.t != Value::STRING) {
  ------------------
  |  Branch (2543:33): [True: 1, False: 171]
  ------------------
 2544|      1|                                throw makeError(ast.location,
 2545|      1|                                                "the left hand side of the 'in' operator should be "
 2546|      1|                                                "a string,  got " +
 2547|      1|                                                    type_str(lhs));
 2548|      1|                            }
 2549|    171|                            auto *field = static_cast<HeapString *>(lhs.v.h);
 2550|    171|                            switch (rhs.t) {
 2551|    170|                                case Value::OBJECT: {
  ------------------
  |  Branch (2551:33): [True: 170, False: 1]
  ------------------
 2552|    170|                                    auto *obj = static_cast<HeapObject *>(rhs.v.h);
 2553|    170|                                    auto *fid = alloc->makeIdentifier(field->value);
 2554|    170|                                    unsigned unused_found_at = 0;
 2555|    170|                                    bool in = findObject(fid, obj, 0, unused_found_at);
 2556|    170|                                    scratch = makeBoolean(in);
 2557|    170|                                } break;
 2558|       |
 2559|      1|                                default:
  ------------------
  |  Branch (2559:33): [True: 1, False: 170]
  ------------------
 2560|      1|                                    throw makeError(
 2561|      1|                                        ast.location,
 2562|      1|                                        "the right hand side of the 'in' operator should be"
 2563|      1|                                        " an object, got " +
 2564|      1|                                            type_str(rhs));
 2565|    171|                            }
 2566|    170|                            goto popframe;
 2567|    171|                        }
 2568|       |
 2569|  2.36M|                        default:;
  ------------------
  |  Branch (2569:25): [True: 2.36M, False: 172]
  ------------------
 2570|  2.36M|                    }
 2571|       |                    // Everything else requires matching types.
 2572|  2.36M|                    if (lhs.t != rhs.t) {
  ------------------
  |  Branch (2572:25): [True: 108, False: 2.36M]
  ------------------
 2573|    108|                        throw makeError(ast.location,
 2574|    108|                                        "binary operator " + bop_string(ast.op) +
 2575|    108|                                            " requires "
 2576|    108|                                            "matching types, got " +
 2577|    108|                                            type_str(lhs) + " and " + type_str(rhs) + ".");
 2578|    108|                    }
 2579|  2.36M|                    switch (lhs.t) {
  ------------------
  |  Branch (2579:29): [True: 2.36M, False: 0]
  ------------------
 2580|   397k|                        case Value::ARRAY:
  ------------------
  |  Branch (2580:25): [True: 397k, False: 1.96M]
  ------------------
 2581|   397k|                            if (ast.op == BOP_PLUS) {
  ------------------
  |  Branch (2581:33): [True: 387k, False: 10.3k]
  ------------------
 2582|   387k|                                auto *arr_l = static_cast<HeapArray *>(lhs.v.h);
 2583|   387k|                                auto *arr_r = static_cast<HeapArray *>(rhs.v.h);
 2584|   387k|                                std::vector<HeapThunk *> elements;
 2585|   387k|                                for (auto *el : arr_l->elements)
  ------------------
  |  Branch (2585:47): [True: 13.1M, False: 387k]
  ------------------
 2586|  13.1M|                                    elements.push_back(el);
 2587|   387k|                                for (auto *el : arr_r->elements)
  ------------------
  |  Branch (2587:47): [True: 346k, False: 387k]
  ------------------
 2588|   346k|                                    elements.push_back(el);
 2589|   387k|                                scratch = makeArray(elements);
 2590|   387k|                            } else if (ast.op == BOP_LESS || ast.op == BOP_LESS_EQ || ast.op == BOP_GREATER || ast.op == BOP_GREATER_EQ) {
  ------------------
  |  Branch (2590:40): [True: 5.83k, False: 4.48k]
  |  Branch (2590:62): [True: 1.45k, False: 3.03k]
  |  Branch (2590:87): [True: 850, False: 2.18k]
  |  Branch (2590:112): [True: 2.18k, False: 6]
  ------------------
 2591|  10.3k|                                HeapThunk *func;
 2592|  10.3k|                                switch(ast.op) {
 2593|  5.83k|                                case BOP_LESS:
  ------------------
  |  Branch (2593:33): [True: 5.83k, False: 4.48k]
  ------------------
 2594|  5.83k|                                    func = sourceVals["__array_less"];
 2595|  5.83k|                                    break;
 2596|  1.45k|                                case BOP_LESS_EQ:
  ------------------
  |  Branch (2596:33): [True: 1.45k, False: 8.86k]
  ------------------
 2597|  1.45k|                                    func = sourceVals["__array_less_or_equal"];
 2598|  1.45k|                                    break;
 2599|    850|                                case BOP_GREATER:
  ------------------
  |  Branch (2599:33): [True: 850, False: 9.46k]
  ------------------
 2600|    850|                                    func = sourceVals["__array_greater"];
 2601|    850|                                    break;
 2602|  2.18k|                                case BOP_GREATER_EQ:
  ------------------
  |  Branch (2602:33): [True: 2.18k, False: 8.13k]
  ------------------
 2603|  2.18k|                                    func = sourceVals["__array_greater_or_equal"];
 2604|  2.18k|                                    break;
 2605|      0|                                default:
  ------------------
  |  Branch (2605:33): [True: 0, False: 10.3k]
  ------------------
 2606|      0|                                    JSONNET_UNREACHABLE();
  ------------------
  |  |   53|      0|  do {                                                             \
  |  |   54|      0|    std::cerr << __FILE__ << ":" << __LINE__                       \
  |  |   55|      0|              << ": INTERNAL ERROR: reached unreachable code path" \
  |  |   56|      0|              << std::endl;                                        \
  |  |   57|      0|    abort();                                                       \
  |  |   58|      0|  } while (0)
  |  |  ------------------
  |  |  |  Branch (58:12): [Folded, False: 0]
  |  |  ------------------
  ------------------
 2607|  10.3k|                                }
 2608|  10.3k|                                if (!func->filled) {
  ------------------
  |  Branch (2608:37): [True: 113, False: 10.2k]
  ------------------
 2609|    113|                                    stack.newCall(ast.location, func, func->self, func->offset, func->upValues);
 2610|    113|                                    ast_ = func->body;
 2611|    113|                                    goto recurse;
 2612|    113|                                }
 2613|  10.2k|                                auto *lhs_th = makeHeap<HeapThunk>(idInternal, f.self, f.offset, ast.left);
 2614|  10.2k|                                lhs_th->fill(lhs);
 2615|  10.2k|                                f.thunks.push_back(lhs_th);
 2616|  10.2k|                                auto *rhs_th = makeHeap<HeapThunk>(idInternal, f.self, f.offset, ast.right);
 2617|  10.2k|                                rhs_th->fill(rhs);
 2618|  10.2k|                                f.thunks.push_back(rhs_th);
 2619|  10.2k|                                const AST *orig_ast = ast_;
 2620|  10.2k|                                stack.pop();
 2621|  10.2k|                                ast_ = callSourceVal(orig_ast, func, {lhs_th, rhs_th});
 2622|  10.2k|                                goto recurse;
 2623|  10.3k|                            } else {
 2624|      6|                                throw makeError(ast.location,
 2625|      6|                                                "binary operator " + bop_string(ast.op) +
 2626|      6|                                                    " does not operate on arrays.");
 2627|      6|                            }
 2628|   387k|                            break;
 2629|       |
 2630|   387k|                        case Value::BOOLEAN:
  ------------------
  |  Branch (2630:25): [True: 163k, False: 2.20M]
  ------------------
 2631|   163k|                            switch (ast.op) {
 2632|  76.2k|                                case BOP_AND: scratch = makeBoolean(lhs.v.b && rhs.v.b); break;
  ------------------
  |  Branch (2632:33): [True: 76.2k, False: 87.0k]
  |  Branch (2632:69): [True: 76.2k, False: 0]
  |  Branch (2632:80): [True: 72.9k, False: 3.32k]
  ------------------
 2633|       |
 2634|  86.9k|                                case BOP_OR: scratch = makeBoolean(lhs.v.b || rhs.v.b); break;
  ------------------
  |  Branch (2634:33): [True: 86.9k, False: 76.3k]
  |  Branch (2634:68): [True: 0, False: 86.9k]
  |  Branch (2634:79): [True: 971, False: 85.9k]
  ------------------
 2635|       |
 2636|     32|                                default:
  ------------------
  |  Branch (2636:33): [True: 32, False: 163k]
  ------------------
 2637|     32|                                    throw makeError(ast.location,
 2638|     32|                                                    "binary operator " + bop_string(ast.op) +
 2639|     32|                                                        " does not operate on booleans.");
 2640|   163k|                            }
 2641|   163k|                            break;
 2642|       |
 2643|   830k|                        case Value::NUMBER:
  ------------------
  |  Branch (2643:25): [True: 830k, False: 1.53M]
  ------------------
 2644|   830k|                            switch (ast.op) {
 2645|   233k|                                case BOP_PLUS:
  ------------------
  |  Branch (2645:33): [True: 233k, False: 596k]
  ------------------
 2646|   233k|                                    scratch = makeNumberCheck(ast.location, lhs.v.d + rhs.v.d);
 2647|   233k|                                    break;
 2648|       |
 2649|  69.9k|                                case BOP_MINUS:
  ------------------
  |  Branch (2649:33): [True: 69.9k, False: 760k]
  ------------------
 2650|  69.9k|                                    scratch = makeNumberCheck(ast.location, lhs.v.d - rhs.v.d);
 2651|  69.9k|                                    break;
 2652|       |
 2653|  2.73k|                                case BOP_MULT:
  ------------------
  |  Branch (2653:33): [True: 2.73k, False: 827k]
  ------------------
 2654|  2.73k|                                    scratch = makeNumberCheck(ast.location, lhs.v.d * rhs.v.d);
 2655|  2.73k|                                    break;
 2656|       |
 2657|  12.9k|                                case BOP_DIV:
  ------------------
  |  Branch (2657:33): [True: 12.9k, False: 817k]
  ------------------
 2658|  12.9k|                                    if (rhs.v.d == 0)
  ------------------
  |  Branch (2658:41): [True: 4, False: 12.9k]
  ------------------
 2659|      4|                                        throw makeError(ast.location, "division by zero.");
 2660|  12.9k|                                    scratch = makeNumberCheck(ast.location, lhs.v.d / rhs.v.d);
 2661|  12.9k|                                    break;
 2662|       |
 2663|       |                                    // No need to check doubles made from longs
 2664|       |
 2665|  1.29k|                                case BOP_SHIFT_L: {
  ------------------
  |  Branch (2665:33): [True: 1.29k, False: 828k]
  ------------------
 2666|  1.29k|                                    if (rhs.v.d < 0)
  ------------------
  |  Branch (2666:41): [True: 1, False: 1.29k]
  ------------------
 2667|      1|                                        throw makeError(ast.location, "shift by negative exponent.");
 2668|       |
 2669|  1.29k|                                    int64_t long_l = safeDoubleToInt64(lhs.v.d, ast.location);
 2670|  1.29k|                                    int64_t long_r = safeDoubleToInt64(rhs.v.d, ast.location);
 2671|  1.29k|                                    long_r = long_r % 64;
 2672|       |
 2673|       |                                    // Additional safety check for left shifts to prevent undefined behavior.
 2674|       |                                    // Left-shift that would move the highest set bit into the sign bit position is undefined.
 2675|  1.29k|                                    if (long_r >= 1 && abs(long_l) >= (1LL << (63 - long_r))) {
  ------------------
  |  Branch (2675:41): [True: 1.08k, False: 204]
  |  Branch (2675:56): [True: 41, False: 1.04k]
  ------------------
 2676|     41|                                        throw makeError(ast.location,
 2677|     41|                                            "numeric value outside safe integer range for bitwise operation.");
 2678|     41|                                    }
 2679|       |
 2680|       |                                    // Left-shift of a negative number is undefined until C++20.
 2681|       |                                    // Perform the shift on unsigned int to avoid that.
 2682|  1.25k|                                    scratch = makeNumber(static_cast<int64_t>(static_cast<uint64_t>(long_l) << long_r));
 2683|  1.25k|                                } break;
 2684|       |
 2685|     37|                                case BOP_SHIFT_R: {
  ------------------
  |  Branch (2685:33): [True: 37, False: 830k]
  ------------------
 2686|     37|                                    if (rhs.v.d < 0)
  ------------------
  |  Branch (2686:41): [True: 1, False: 36]
  ------------------
 2687|      1|                                        throw makeError(ast.location, "shift by negative exponent.");
 2688|       |
 2689|     36|                                    int64_t long_l = safeDoubleToInt64(lhs.v.d, ast.location);
 2690|     36|                                    int64_t long_r = safeDoubleToInt64(rhs.v.d, ast.location);
 2691|     36|                                    long_r = long_r % 64;
 2692|     36|                                    scratch = makeNumber(long_l >> long_r);
 2693|     36|                                } break;
 2694|       |
 2695|  4.07k|                                case BOP_BITWISE_AND: {
  ------------------
  |  Branch (2695:33): [True: 4.07k, False: 826k]
  ------------------
 2696|  4.07k|                                    int64_t long_l = safeDoubleToInt64(lhs.v.d, ast.location);
 2697|  4.07k|                                    int64_t long_r = safeDoubleToInt64(rhs.v.d, ast.location);
 2698|  4.07k|                                    scratch = makeNumber(long_l & long_r);
 2699|  4.07k|                                } break;
 2700|       |
 2701|  1.90k|                                case BOP_BITWISE_XOR: {
  ------------------
  |  Branch (2701:33): [True: 1.90k, False: 828k]
  ------------------
 2702|  1.90k|                                    int64_t long_l = safeDoubleToInt64(lhs.v.d, ast.location);
 2703|  1.90k|                                    int64_t long_r = safeDoubleToInt64(rhs.v.d, ast.location);
 2704|  1.90k|                                    scratch = makeNumber(long_l ^ long_r);
 2705|  1.90k|                                } break;
 2706|       |
 2707|    288|                                case BOP_BITWISE_OR: {
  ------------------
  |  Branch (2707:33): [True: 288, False: 829k]
  ------------------
 2708|    288|                                    int64_t long_l = safeDoubleToInt64(lhs.v.d, ast.location);
 2709|    288|                                    int64_t long_r = safeDoubleToInt64(rhs.v.d, ast.location);
 2710|    288|                                    scratch = makeNumber(long_l | long_r);
 2711|    288|                                } break;
 2712|       |
 2713|  38.6k|                                case BOP_LESS_EQ: scratch = makeBoolean(lhs.v.d <= rhs.v.d); break;
  ------------------
  |  Branch (2713:33): [True: 38.6k, False: 791k]
  ------------------
 2714|       |
 2715|   377k|                                case BOP_GREATER_EQ:
  ------------------
  |  Branch (2715:33): [True: 377k, False: 452k]
  ------------------
 2716|   377k|                                    scratch = makeBoolean(lhs.v.d >= rhs.v.d);
 2717|   377k|                                    break;
 2718|       |
 2719|  70.8k|                                case BOP_LESS: scratch = makeBoolean(lhs.v.d < rhs.v.d); break;
  ------------------
  |  Branch (2719:33): [True: 70.8k, False: 759k]
  ------------------
 2720|       |
 2721|  15.6k|                                case BOP_GREATER: scratch = makeBoolean(lhs.v.d > rhs.v.d); break;
  ------------------
  |  Branch (2721:33): [True: 15.6k, False: 814k]
  ------------------
 2722|       |
 2723|      5|                                default:
  ------------------
  |  Branch (2723:33): [True: 5, False: 830k]
  ------------------
 2724|      5|                                    throw makeError(ast.location,
 2725|      5|                                                    "binary operator " + bop_string(ast.op) +
 2726|      5|                                                        " does not operate on numbers.");
 2727|   830k|                            }
 2728|   830k|                            break;
 2729|       |
 2730|   830k|                        case Value::FUNCTION:
  ------------------
  |  Branch (2730:25): [True: 0, False: 2.36M]
  ------------------
 2731|      0|                            throw makeError(ast.location,
 2732|      0|                                            "binary operator " + bop_string(ast.op) +
 2733|      0|                                                " does not operate on functions.");
 2734|       |
 2735|      2|                        case Value::NULL_TYPE:
  ------------------
  |  Branch (2735:25): [True: 2, False: 2.36M]
  ------------------
 2736|      2|                            throw makeError(ast.location,
 2737|      2|                                            "binary operator " + bop_string(ast.op) +
 2738|      2|                                                " does not operate on null.");
 2739|       |
 2740|   881k|                        case Value::OBJECT: {
  ------------------
  |  Branch (2740:25): [True: 881k, False: 1.48M]
  ------------------
 2741|   881k|                            if (ast.op != BOP_PLUS) {
  ------------------
  |  Branch (2741:33): [True: 7, False: 881k]
  ------------------
 2742|      7|                                throw makeError(ast.location,
 2743|      7|                                                "binary operator " + bop_string(ast.op) +
 2744|      7|                                                    " does not operate on objects.");
 2745|      7|                            }
 2746|   881k|                            auto *lhs_obj = static_cast<HeapObject *>(lhs.v.h);
 2747|   881k|                            auto *rhs_obj = static_cast<HeapObject *>(rhs.v.h);
 2748|   881k|                            scratch = makeObject<HeapExtendedObject>(lhs_obj, rhs_obj);
 2749|   881k|                        } break;
 2750|       |
 2751|  90.8k|                        case Value::STRING: {
  ------------------
  |  Branch (2751:25): [True: 90.8k, False: 2.27M]
  ------------------
 2752|  90.8k|                            const UString &lhs_str = static_cast<HeapString *>(lhs.v.h)->value;
 2753|  90.8k|                            const UString &rhs_str = static_cast<HeapString *>(rhs.v.h)->value;
 2754|  90.8k|                            switch (ast.op) {
 2755|      0|                                case BOP_PLUS: scratch = makeString(lhs_str + rhs_str); break;
  ------------------
  |  Branch (2755:33): [True: 0, False: 90.8k]
  ------------------
 2756|       |
 2757|  7.68k|                                case BOP_LESS_EQ: scratch = makeBoolean(lhs_str <= rhs_str); break;
  ------------------
  |  Branch (2757:33): [True: 7.68k, False: 83.1k]
  ------------------
 2758|       |
 2759|  71.2k|                                case BOP_GREATER_EQ:
  ------------------
  |  Branch (2759:33): [True: 71.2k, False: 19.6k]
  ------------------
 2760|  71.2k|                                    scratch = makeBoolean(lhs_str >= rhs_str);
 2761|  71.2k|                                    break;
 2762|       |
 2763|  5.54k|                                case BOP_LESS: scratch = makeBoolean(lhs_str < rhs_str); break;
  ------------------
  |  Branch (2763:33): [True: 5.54k, False: 85.2k]
  ------------------
 2764|       |
 2765|  6.36k|                                case BOP_GREATER: scratch = makeBoolean(lhs_str > rhs_str); break;
  ------------------
  |  Branch (2765:33): [True: 6.36k, False: 84.4k]
  ------------------
 2766|       |
 2767|     16|                                default:
  ------------------
  |  Branch (2767:33): [True: 16, False: 90.8k]
  ------------------
 2768|     16|                                    throw makeError(ast.location,
 2769|     16|                                                    "binary operator " + bop_string(ast.op) +
 2770|     16|                                                        " does not operate on strings.");
 2771|  90.8k|                            }
 2772|  90.8k|                        } break;
 2773|  2.36M|                    }
 2774|  2.36M|                } break;
 2775|       |
 2776|  2.35M|                case FRAME_BUILTIN_FILTER: {
  ------------------
  |  Branch (2776:17): [True: 0, False: 106M]
  ------------------
 2777|      0|                    const auto &ast = *static_cast<const Apply *>(f.ast);
 2778|      0|                    auto *func = static_cast<HeapClosure *>(f.val.v.h);
 2779|      0|                    auto *arr = static_cast<HeapArray *>(f.val2.v.h);
 2780|      0|                    if (scratch.t != Value::BOOLEAN) {
  ------------------
  |  Branch (2780:25): [True: 0, False: 0]
  ------------------
 2781|      0|                        throw makeError(
 2782|      0|                            ast.location,
 2783|      0|                            "filter function must return boolean, got: " + type_str(scratch));
 2784|      0|                    }
 2785|      0|                    if (scratch.v.b)
  ------------------
  |  Branch (2785:25): [True: 0, False: 0]
  ------------------
 2786|      0|                        f.thunks.push_back(arr->elements[f.elementId]);
 2787|      0|                    f.elementId++;
 2788|       |                    // Iterate through arr, calling the function on each.
 2789|      0|                    if (f.elementId == arr->elements.size()) {
  ------------------
  |  Branch (2789:25): [True: 0, False: 0]
  ------------------
 2790|      0|                        scratch = makeArray(f.thunks);
 2791|      0|                    } else {
 2792|      0|                        auto *thunk = arr->elements[f.elementId];
 2793|      0|                        BindingFrame bindings = func->upValues;
 2794|      0|                        bindings[func->params[0].id] = thunk;
 2795|      0|                        stack.newCall(ast.location, func, func->self, func->offset, bindings);
 2796|      0|                        ast_ = func->body;
 2797|      0|                        goto recurse;
 2798|      0|                    }
 2799|      0|                } break;
 2800|       |
 2801|  16.7M|                case FRAME_BUILTIN_FORCE_THUNKS: {
  ------------------
  |  Branch (2801:17): [True: 16.7M, False: 89.5M]
  ------------------
 2802|  16.7M|                    const auto &location = f.location;
 2803|  16.7M|                    auto *func = static_cast<HeapClosure *>(f.val.v.h);
 2804|  16.7M|                    if (f.elementId == f.thunks.size()) {
  ------------------
  |  Branch (2804:25): [True: 6.49M, False: 10.2M]
  ------------------
 2805|       |                        // All thunks forced, now the builtin implementations.
 2806|  6.49M|                        const std::string &builtin_name = func->builtinName;
 2807|  6.49M|                        std::vector<Value> args;
 2808|  10.2M|                        for (const auto &p : func->params) {
  ------------------
  |  Branch (2808:44): [True: 10.2M, False: 6.49M]
  ------------------
 2809|  10.2M|                            const auto it = f.bindings.find(p.id);
 2810|  10.2M|                            if (it != f.bindings.end()) {
  ------------------
  |  Branch (2810:33): [True: 10.2M, False: 0]
  ------------------
 2811|  10.2M|                                assert(it->second->filled);
  ------------------
  |  Branch (2811:33): [True: 10.2M, False: 0]
  ------------------
 2812|  10.2M|                                args.push_back(it->second->content);
 2813|  10.2M|                            } else {
 2814|      0|                                std::stringstream ss;
 2815|      0|                                ss << "function parameter " << encode_utf8(p.id->name)
 2816|      0|                                    << " not bound in call.";
 2817|      0|                                throw makeError(location, ss.str());
 2818|      0|                            }
 2819|  10.2M|                        }
 2820|       |
 2821|  6.49M|                        BuiltinMap::const_iterator bit = builtins.find(builtin_name);
 2822|  6.49M|                        if (bit != builtins.end()) {
  ------------------
  |  Branch (2822:29): [True: 6.49M, False: 0]
  ------------------
 2823|  6.49M|                            const AST *new_ast = (this->*bit->second)(location, args);
 2824|  6.49M|                            if (new_ast != nullptr) {
  ------------------
  |  Branch (2824:33): [True: 0, False: 6.49M]
  ------------------
 2825|      0|                                ast_ = new_ast;
 2826|      0|                                goto recurse;
 2827|      0|                            }
 2828|  6.49M|                            break;
 2829|  6.49M|                        }
 2830|      0|                        VmNativeCallbackMap::const_iterator nit =
 2831|      0|                            nativeCallbacks.find(builtin_name);
 2832|       |                        // TODO(dcunnin): Support arrays.
 2833|       |                        // TODO(dcunnin): Support objects.
 2834|      0|                        std::vector<JsonnetJsonValue> args2;
 2835|      0|                        for (const Value &arg : args) {
  ------------------
  |  Branch (2835:47): [True: 0, False: 0]
  ------------------
 2836|      0|                            switch (arg.t) {
 2837|      0|                                case Value::STRING:
  ------------------
  |  Branch (2837:33): [True: 0, False: 0]
  ------------------
 2838|      0|                                    args2.emplace_back(
 2839|      0|                                        JsonnetJsonValue::STRING,
 2840|      0|                                        encode_utf8(static_cast<HeapString *>(arg.v.h)->value),
 2841|      0|                                        0);
 2842|      0|                                    break;
 2843|       |
 2844|      0|                                case Value::BOOLEAN:
  ------------------
  |  Branch (2844:33): [True: 0, False: 0]
  ------------------
 2845|      0|                                    args2.emplace_back(
 2846|      0|                                        JsonnetJsonValue::BOOL, "", arg.v.b ? 1.0 : 0.0);
  ------------------
  |  Branch (2846:69): [True: 0, False: 0]
  ------------------
 2847|      0|                                    break;
 2848|       |
 2849|      0|                                case Value::NUMBER:
  ------------------
  |  Branch (2849:33): [True: 0, False: 0]
  ------------------
 2850|      0|                                    args2.emplace_back(JsonnetJsonValue::NUMBER, "", arg.v.d);
 2851|      0|                                    break;
 2852|       |
 2853|      0|                                case Value::NULL_TYPE:
  ------------------
  |  Branch (2853:33): [True: 0, False: 0]
  ------------------
 2854|      0|                                    args2.emplace_back(JsonnetJsonValue::NULL_KIND, "", 0);
 2855|      0|                                    break;
 2856|       |
 2857|      0|                                default:
  ------------------
  |  Branch (2857:33): [True: 0, False: 0]
  ------------------
 2858|      0|                                    throw makeError(location,
 2859|      0|                                                    "native extensions can only take primitives.");
 2860|      0|                            }
 2861|      0|                        }
 2862|      0|                        std::vector<const JsonnetJsonValue *> args3;
 2863|      0|                        for (size_t i = 0; i < args2.size(); ++i) {
  ------------------
  |  Branch (2863:44): [True: 0, False: 0]
  ------------------
 2864|      0|                            args3.push_back(&args2[i]);
 2865|      0|                        }
 2866|      0|                        if (nit == nativeCallbacks.end()) {
  ------------------
  |  Branch (2866:29): [True: 0, False: 0]
  ------------------
 2867|      0|                            throw makeError(location,
 2868|      0|                                            "unrecognized builtin name: " + builtin_name);
 2869|      0|                        }
 2870|      0|                        const VmNativeCallback &cb = nit->second;
 2871|       |
 2872|      0|                        int succ;
 2873|      0|                        std::unique_ptr<JsonnetJsonValue> r(cb.cb(cb.ctx, args3.data(), &succ));
 2874|       |
 2875|      0|                        if (succ) {
  ------------------
  |  Branch (2875:29): [True: 0, False: 0]
  ------------------
 2876|      0|                            bool unused;
 2877|      0|                            jsonToHeap(r, unused, scratch);
 2878|      0|                        } else {
 2879|      0|                            if (r->kind != JsonnetJsonValue::STRING) {
  ------------------
  |  Branch (2879:33): [True: 0, False: 0]
  ------------------
 2880|      0|                                throw makeError(
 2881|      0|                                    location,
 2882|      0|                                    "native extension returned an error that was not a string.");
 2883|      0|                            }
 2884|      0|                            std::string rs = r->string;
 2885|      0|                            throw makeError(location, rs);
 2886|      0|                        }
 2887|       |
 2888|  10.2M|                    } else {
 2889|       |                        // Not all arguments forced yet.
 2890|  10.2M|                        HeapThunk *th = f.thunks[f.elementId++];
 2891|  10.2M|                        if (!th->filled) {
  ------------------
  |  Branch (2891:29): [True: 10.2M, False: 0]
  ------------------
 2892|  10.2M|                            stack.newCall(location, th, th->self, th->offset, th->upValues);
 2893|  10.2M|                            ast_ = th->body;
 2894|  10.2M|                            goto recurse;
 2895|  10.2M|                        } else {
 2896|       |                            // Otherwise loop back around to force the next thunk.
 2897|      0|                            goto replaceframe;
 2898|      0|                        }
 2899|  10.2M|                    }
 2900|  16.7M|                } break;
 2901|       |
 2902|  33.1M|                case FRAME_CALL: {
  ------------------
  |  Branch (2902:17): [True: 33.1M, False: 73.1M]
  ------------------
 2903|  33.1M|                    if (auto *thunk = dynamic_cast<HeapThunk *>(f.context)) {
  ------------------
  |  Branch (2903:31): [True: 18.2M, False: 14.9M]
  ------------------
 2904|       |                        // If we called a thunk, cache result.
 2905|  18.2M|                        thunk->fill(scratch);
 2906|  18.2M|                    } else if (auto *closure = dynamic_cast<HeapClosure *>(f.context)) {
  ------------------
  |  Branch (2906:38): [True: 5.47M, False: 9.47M]
  ------------------
 2907|  5.47M|                        if (f.elementId < f.thunks.size()) {
  ------------------
  |  Branch (2907:29): [True: 2.69M, False: 2.78M]
  ------------------
 2908|       |                            // If tailstrict, force thunks
 2909|  2.69M|                            HeapThunk *th = f.thunks[f.elementId++];
 2910|  2.69M|                            if (!th->filled) {
  ------------------
  |  Branch (2910:33): [True: 2.69M, False: 0]
  ------------------
 2911|  2.69M|                                stack.newCall(f.location, th, th->self, th->offset, th->upValues);
 2912|  2.69M|                                ast_ = th->body;
 2913|  2.69M|                                goto recurse;
 2914|  2.69M|                            }
 2915|  2.78M|                        } else if (f.thunks.size() == 0) {
  ------------------
  |  Branch (2915:36): [True: 1.56M, False: 1.22M]
  ------------------
 2916|       |                            // Body has now been executed
 2917|  1.56M|                        } else {
 2918|       |                            // Execute the body
 2919|  1.22M|                            f.thunks.clear();
 2920|  1.22M|                            f.elementId = 0;
 2921|  1.22M|                            ast_ = closure->body;
 2922|  1.22M|                            goto recurse;
 2923|  1.22M|                        }
 2924|  5.47M|                    }
 2925|       |                    // Result of call is in scratch, just pop.
 2926|  33.1M|                } break;
 2927|       |
 2928|  29.2M|                case FRAME_ERROR: {
  ------------------
  |  Branch (2928:17): [True: 970, False: 106M]
  ------------------
 2929|    970|                    const auto &ast = *static_cast<const Error *>(f.ast);
 2930|    970|                    UString msg;
 2931|    970|                    if (scratch.t == Value::STRING) {
  ------------------
  |  Branch (2931:25): [True: 395, False: 575]
  ------------------
 2932|    395|                        msg = static_cast<HeapString *>(scratch.v.h)->value;
 2933|    575|                    } else {
 2934|    575|                        msg = toString(ast.location);
 2935|    575|                    }
 2936|    970|                    throw makeError(ast.location, encode_utf8(msg));
 2937|  33.1M|                } break;
 2938|       |
 2939|  5.29M|                case FRAME_IF: {
  ------------------
  |  Branch (2939:17): [True: 5.29M, False: 101M]
  ------------------
 2940|  5.29M|                    const auto &ast = *static_cast<const Conditional *>(f.ast);
 2941|  5.29M|                    if (scratch.t != Value::BOOLEAN) {
  ------------------
  |  Branch (2941:25): [True: 47, False: 5.29M]
  ------------------
 2942|     47|                        throw makeError(
 2943|     47|                            ast.location,
 2944|     47|                            "condition must be boolean, got " + type_str(scratch) + ".");
 2945|     47|                    }
 2946|  5.29M|                    ast_ = scratch.v.b ? ast.branchTrue : ast.branchFalse;
  ------------------
  |  Branch (2946:28): [True: 1.86M, False: 3.42M]
  ------------------
 2947|  5.29M|                    stack.pop();
 2948|  5.29M|                    goto recurse;
 2949|  5.29M|                } break;
 2950|       |
 2951|  1.16M|                case FRAME_SUPER_INDEX: {
  ------------------
  |  Branch (2951:17): [True: 1.16M, False: 105M]
  ------------------
 2952|  1.16M|                    const auto &ast = *static_cast<const SuperIndex *>(f.ast);
 2953|  1.16M|                    HeapObject *self;
 2954|  1.16M|                    unsigned offset;
 2955|  1.16M|                    stack.getSelfBinding(self, offset);
 2956|  1.16M|                    offset++;
 2957|  1.16M|                    if (offset >= countLeaves(self)) {
  ------------------
  |  Branch (2957:25): [True: 2, False: 1.16M]
  ------------------
 2958|      2|                        throw makeError(ast.location,
 2959|      2|                                        "attempt to use super when there is no super class.");
 2960|      2|                    }
 2961|  1.16M|                    if (scratch.t != Value::STRING) {
  ------------------
  |  Branch (2961:25): [True: 0, False: 1.16M]
  ------------------
 2962|      0|                        throw makeError(
 2963|      0|                            ast.location,
 2964|      0|                            "super index must be string, got " + type_str(scratch) + ".");
 2965|      0|                    }
 2966|       |
 2967|  1.16M|                    const UString &index_name = static_cast<HeapString *>(scratch.v.h)->value;
 2968|  1.16M|                    auto *fid = alloc->makeIdentifier(index_name);
 2969|  1.16M|                    stack.pop();
 2970|  1.16M|                    ast_ = objectIndex(ast.location, self, fid, offset);
 2971|  1.16M|                    goto recurse;
 2972|  1.16M|                } break;
 2973|       |
 2974|  1.32M|                case FRAME_IN_SUPER_ELEMENT: {
  ------------------
  |  Branch (2974:17): [True: 1.32M, False: 105M]
  ------------------
 2975|  1.32M|                    const auto &ast = *static_cast<const InSuper *>(f.ast);
 2976|  1.32M|                    HeapObject *self;
 2977|  1.32M|                    unsigned offset;
 2978|  1.32M|                    stack.getSelfBinding(self, offset);
 2979|  1.32M|                    offset++;
 2980|  1.32M|                    if (scratch.t != Value::STRING) {
  ------------------
  |  Branch (2980:25): [True: 0, False: 1.32M]
  ------------------
 2981|      0|                        throw makeError(ast.location,
 2982|      0|                                        "left hand side of e in super must be string, got " +
 2983|      0|                                            type_str(scratch) + ".");
 2984|      0|                    }
 2985|  1.32M|                    if (offset >= countLeaves(self)) {
  ------------------
  |  Branch (2985:25): [True: 52.6k, False: 1.27M]
  ------------------
 2986|       |                        // There is no super object.
 2987|  52.6k|                        scratch = makeBoolean(false);
 2988|  1.27M|                    } else {
 2989|  1.27M|                        const UString &element_name = static_cast<HeapString *>(scratch.v.h)->value;
 2990|  1.27M|                        auto *fid = alloc->makeIdentifier(element_name);
 2991|  1.27M|                        unsigned unused_found_at = 0;
 2992|  1.27M|                        bool in = findObject(fid, self, offset, unused_found_at);
 2993|  1.27M|                        scratch = makeBoolean(in);
 2994|  1.27M|                    }
 2995|  1.32M|                } break;
 2996|       |
 2997|  8.69M|                case FRAME_INDEX_INDEX: {
  ------------------
  |  Branch (2997:17): [True: 8.69M, False: 97.6M]
  ------------------
 2998|  8.69M|                    const auto &ast = *static_cast<const Index *>(f.ast);
 2999|  8.69M|                    const Value &target = f.val;
 3000|  8.69M|                    if (target.t == Value::ARRAY) {
  ------------------
  |  Branch (3000:25): [True: 185k, False: 8.51M]
  ------------------
 3001|   185k|                        const auto *array = static_cast<HeapArray *>(target.v.h);
 3002|   185k|                        if (scratch.t == Value::STRING) {
  ------------------
  |  Branch (3002:29): [True: 37, False: 184k]
  ------------------
 3003|     37|                            const UString &str = static_cast<HeapString *>(scratch.v.h)->value;
 3004|     37|                            throw makeError(
 3005|     37|                                ast.location,
 3006|     37|                                "attempted index an array with string \""
 3007|     37|                                + encode_utf8(jsonnet_string_escape(str, false)) + "\".");
 3008|     37|                        }
 3009|   184k|                        if (scratch.t != Value::NUMBER) {
  ------------------
  |  Branch (3009:29): [True: 12, False: 184k]
  ------------------
 3010|     12|                            throw makeError(
 3011|     12|                                ast.location,
 3012|     12|                                "array index must be number, got " + type_str(scratch) + ".");
 3013|     12|                        }
 3014|   184k|                        double index = ::floor(scratch.v.d);
 3015|   184k|                        long sz = array->elements.size();
 3016|   184k|                        if (index < 0 || index >= sz) {
  ------------------
  |  Branch (3016:29): [True: 1, False: 184k]
  |  Branch (3016:42): [True: 7, False: 184k]
  ------------------
 3017|      8|                            std::stringstream ss;
 3018|      8|                            ss << "array bounds error: " << index << " not within [0, " << sz
 3019|      8|                               << ")";
 3020|      8|                            throw makeError(ast.location, ss.str());
 3021|      8|                        }
 3022|   184k|                        if (scratch.v.d != index) {
  ------------------
  |  Branch (3022:29): [True: 8, False: 184k]
  ------------------
 3023|      8|                            std::stringstream ss;
 3024|      8|                            ss << "array index was not integer: " << scratch.v.d;
 3025|      8|                            throw makeError(ast.location, ss.str());
 3026|      8|                        }
 3027|       |                        // index < sz <= SIZE_T_MAX
 3028|   184k|                        auto *thunk = array->elements[size_t(index)];
 3029|   184k|                        if (thunk->filled) {
  ------------------
  |  Branch (3029:29): [True: 71.3k, False: 113k]
  ------------------
 3030|  71.3k|                            scratch = thunk->content;
 3031|   113k|                        } else {
 3032|   113k|                            stack.pop();
 3033|   113k|                            stack.newCall(
 3034|   113k|                                ast.location, thunk, thunk->self, thunk->offset, thunk->upValues);
 3035|   113k|                            ast_ = thunk->body;
 3036|   113k|                            goto recurse;
 3037|   113k|                        }
 3038|  8.51M|                    } else if (target.t == Value::OBJECT) {
  ------------------
  |  Branch (3038:32): [True: 8.35M, False: 156k]
  ------------------
 3039|  8.35M|                        auto *obj = static_cast<HeapObject *>(target.v.h);
 3040|  8.35M|                        assert(obj != nullptr);
  ------------------
  |  Branch (3040:25): [True: 8.35M, False: 0]
  ------------------
 3041|  8.35M|                        if (scratch.t != Value::STRING) {
  ------------------
  |  Branch (3041:29): [True: 5, False: 8.35M]
  ------------------
 3042|      5|                            throw makeError(
 3043|      5|                                ast.location,
 3044|      5|                                "object index must be string, got " + type_str(scratch) + ".");
 3045|      5|                        }
 3046|  8.35M|                        const UString &index_name = static_cast<HeapString *>(scratch.v.h)->value;
 3047|  8.35M|                        auto *fid = alloc->makeIdentifier(index_name);
 3048|  8.35M|                        stack.pop();
 3049|  8.35M|                        ast_ = objectIndex(ast.location, obj, fid, 0);
 3050|  8.35M|                        goto recurse;
 3051|  8.35M|                    } else if (target.t == Value::STRING) {
  ------------------
  |  Branch (3051:32): [True: 156k, False: 0]
  ------------------
 3052|   156k|                        auto *obj = static_cast<HeapString *>(target.v.h);
 3053|   156k|                        assert(obj != nullptr);
  ------------------
  |  Branch (3053:25): [True: 156k, False: 0]
  ------------------
 3054|   156k|                        if (scratch.t != Value::NUMBER) {
  ------------------
  |  Branch (3054:29): [True: 3, False: 156k]
  ------------------
 3055|      3|                            throw makeError(
 3056|      3|                                ast.location,
 3057|      3|                                "string index must be a number, got " + type_str(scratch) + ".");
 3058|      3|                        }
 3059|   156k|                        long sz = obj->value.length();
 3060|   156k|                        long i = (long)scratch.v.d;
 3061|   156k|                        if (i < 0 || i >= sz) {
  ------------------
  |  Branch (3061:29): [True: 54, False: 156k]
  |  Branch (3061:38): [True: 48, False: 155k]
  ------------------
 3062|    102|                            std::stringstream ss;
 3063|    102|                            ss << "string bounds error: " << i << " not within [0, " << sz << ")";
 3064|    102|                            throw makeError(ast.location, ss.str());
 3065|    102|                        }
 3066|   155k|                        char32_t ch[] = {obj->value[i], U'\0'};
 3067|   155k|                        scratch = makeString(ch);
 3068|   155k|                    } else {
 3069|      0|                        std::cerr << "INTERNAL ERROR: not object / array / string." << std::endl;
 3070|      0|                        abort();
 3071|      0|                    }
 3072|  8.69M|                } break;
 3073|       |
 3074|  8.69M|                case FRAME_INDEX_TARGET: {
  ------------------
  |  Branch (3074:17): [True: 8.69M, False: 97.6M]
  ------------------
 3075|  8.69M|                    const auto &ast = *static_cast<const Index *>(f.ast);
 3076|  8.69M|                    if (scratch.t != Value::ARRAY && scratch.t != Value::OBJECT &&
  ------------------
  |  Branch (3076:25): [True: 8.51M, False: 185k]
  |  Branch (3076:54): [True: 156k, False: 8.35M]
  ------------------
 3077|   156k|                        scratch.t != Value::STRING) {
  ------------------
  |  Branch (3077:25): [True: 26, False: 156k]
  ------------------
 3078|     26|                        throw makeError(ast.location,
 3079|     26|                                        "can only index objects, strings, and arrays, got " +
 3080|     26|                                            type_str(scratch) + ".");
 3081|     26|                    }
 3082|  8.69M|                    f.val = scratch;
 3083|  8.69M|                    f.kind = FRAME_INDEX_INDEX;
 3084|  8.69M|                    if (scratch.t == Value::OBJECT) {
  ------------------
  |  Branch (3084:25): [True: 8.35M, False: 341k]
  ------------------
 3085|  8.35M|                        auto *self = static_cast<HeapObject *>(scratch.v.h);
 3086|  8.35M|                        if (!stack.alreadyExecutingInvariants(self)) {
  ------------------
  |  Branch (3086:29): [True: 8.35M, False: 6.50k]
  ------------------
 3087|  8.35M|                            stack.newFrame(FRAME_INVARIANTS, ast.location);
 3088|  8.35M|                            Frame &f2 = stack.top();
 3089|  8.35M|                            f2.self = self;
 3090|  8.35M|                            unsigned counter = 0;
 3091|  8.35M|                            objectInvariants(self, self, counter, f2.thunks);
 3092|  8.35M|                            if (f2.thunks.size() > 0) {
  ------------------
  |  Branch (3092:33): [True: 633, False: 8.35M]
  ------------------
 3093|    633|                                auto *thunk = f2.thunks[0];
 3094|    633|                                f2.elementId = 1;
 3095|    633|                                stack.newCall(ast.location,
 3096|    633|                                              thunk,
 3097|    633|                                              thunk->self,
 3098|    633|                                              thunk->offset,
 3099|    633|                                              thunk->upValues);
 3100|    633|                                ast_ = thunk->body;
 3101|    633|                                goto recurse;
 3102|    633|                            }
 3103|  8.35M|                        }
 3104|  8.35M|                    }
 3105|  8.69M|                    ast_ = ast.index;
 3106|  8.69M|                    goto recurse;
 3107|  8.69M|                } break;
 3108|       |
 3109|  8.39M|                case FRAME_INVARIANTS: {
  ------------------
  |  Branch (3109:17): [True: 8.39M, False: 97.9M]
  ------------------
 3110|  8.39M|                    if (f.elementId >= f.thunks.size()) {
  ------------------
  |  Branch (3110:25): [True: 8.36M, False: 27.3k]
  ------------------
 3111|  8.36M|                        if (stack.size() == initial_stack_size + 1) {
  ------------------
  |  Branch (3111:29): [True: 12.2k, False: 8.35M]
  ------------------
 3112|       |                            // Just pop, evaluate was invoked by runInvariants.
 3113|  12.2k|                            break;
 3114|  12.2k|                        }
 3115|  8.35M|                        stack.pop();
 3116|  8.35M|                        Frame &f2 = stack.top();
 3117|  8.35M|                        const auto &ast = *static_cast<const Index *>(f2.ast);
 3118|  8.35M|                        ast_ = ast.index;
 3119|  8.35M|                        goto recurse;
 3120|  8.36M|                    }
 3121|  27.3k|                    auto *thunk = f.thunks[f.elementId++];
 3122|  27.3k|                    stack.newCall(f.location, thunk, thunk->self, thunk->offset, thunk->upValues);
 3123|  27.3k|                    ast_ = thunk->body;
 3124|  27.3k|                    goto recurse;
 3125|  8.39M|                } break;
 3126|       |
 3127|  3.74M|                case FRAME_LOCAL: {
  ------------------
  |  Branch (3127:17): [True: 3.74M, False: 102M]
  ------------------
 3128|       |                    // Result of execution is in scratch already.
 3129|  3.74M|                } break;
 3130|       |
 3131|  1.77M|                case FRAME_OBJECT: {
  ------------------
  |  Branch (3131:17): [True: 1.77M, False: 104M]
  ------------------
 3132|  1.77M|                    const auto &ast = *static_cast<const DesugaredObject *>(f.ast);
 3133|  1.77M|                    if (scratch.t != Value::NULL_TYPE) {
  ------------------
  |  Branch (3133:25): [True: 1.77M, False: 1]
  ------------------
 3134|  1.77M|                        if (scratch.t != Value::STRING) {
  ------------------
  |  Branch (3134:29): [True: 8, False: 1.77M]
  ------------------
 3135|      8|                            throw makeError(ast.location, "field name was not a string.");
 3136|      8|                        }
 3137|  1.77M|                        const auto &fname = static_cast<const HeapString *>(scratch.v.h)->value;
 3138|  1.77M|                        const Identifier *fid = alloc->makeIdentifier(fname);
 3139|  1.77M|                        if (f.objectFields.find(fid) != f.objectFields.end()) {
  ------------------
  |  Branch (3139:29): [True: 3, False: 1.77M]
  ------------------
 3140|      3|                            std::string msg =
 3141|      3|                                "duplicate field name: \"" + encode_utf8(fname) + "\"";
 3142|      3|                            throw makeError(ast.location, msg);
 3143|      3|                        }
 3144|  1.77M|                        f.objectFields[fid].hide = f.fit->hide;
 3145|  1.77M|                        f.objectFields[fid].body = f.fit->body;
 3146|  1.77M|                    }
 3147|  1.77M|                    f.fit++;
 3148|  1.77M|                    if (f.fit != ast.fields.end()) {
  ------------------
  |  Branch (3148:25): [True: 1.27M, False: 498k]
  ------------------
 3149|  1.27M|                        ast_ = f.fit->name;
 3150|  1.27M|                        goto recurse;
 3151|  1.27M|                    } else {
 3152|   498k|                        auto env = capture(ast.freeVariables);
 3153|   498k|                        scratch = makeObject<HeapSimpleObject>(env, f.objectFields, ast.asserts);
 3154|   498k|                    }
 3155|  1.77M|                } break;
 3156|       |
 3157|   498k|                case FRAME_OBJECT_COMP_ARRAY: {
  ------------------
  |  Branch (3157:17): [True: 0, False: 106M]
  ------------------
 3158|      0|                    const auto &ast = *static_cast<const ObjectComprehensionSimple *>(f.ast);
 3159|      0|                    const Value &arr_v = scratch;
 3160|      0|                    if (scratch.t != Value::ARRAY) {
  ------------------
  |  Branch (3160:25): [True: 0, False: 0]
  ------------------
 3161|      0|                        throw makeError(ast.location,
 3162|      0|                                        "object comprehension needs array, got " + type_str(arr_v));
 3163|      0|                    }
 3164|      0|                    const auto *arr = static_cast<const HeapArray *>(arr_v.v.h);
 3165|      0|                    if (arr->elements.size() == 0) {
  ------------------
  |  Branch (3165:25): [True: 0, False: 0]
  ------------------
 3166|       |                        // Degenerate case.  Just create the object now.
 3167|      0|                        scratch = makeObject<HeapComprehensionObject>(
 3168|      0|                            BindingFrame{}, ast.value, ast.id, BindingFrame{});
 3169|      0|                    } else {
 3170|      0|                        f.kind = FRAME_OBJECT_COMP_ELEMENT;
 3171|      0|                        f.val = scratch;
 3172|      0|                        f.bindings[ast.id] = arr->elements[0];
 3173|      0|                        f.elementId = 0;
 3174|      0|                        ast_ = ast.field;
 3175|      0|                        goto recurse;
 3176|      0|                    }
 3177|      0|                } break;
 3178|       |
 3179|      0|                case FRAME_OBJECT_COMP_ELEMENT: {
  ------------------
  |  Branch (3179:17): [True: 0, False: 106M]
  ------------------
 3180|      0|                    const auto &ast = *static_cast<const ObjectComprehensionSimple *>(f.ast);
 3181|      0|                    const auto *arr = static_cast<const HeapArray *>(f.val.v.h);
 3182|      0|                    if (scratch.t != Value::NULL_TYPE) {
  ------------------
  |  Branch (3182:25): [True: 0, False: 0]
  ------------------
 3183|      0|                        if (scratch.t != Value::STRING) {
  ------------------
  |  Branch (3183:29): [True: 0, False: 0]
  ------------------
 3184|      0|                            std::stringstream ss;
 3185|      0|                            ss << "field must be string, got: " << type_str(scratch);
 3186|      0|                            throw makeError(ast.location, ss.str());
 3187|      0|                        }
 3188|      0|                        const auto &fname = static_cast<const HeapString *>(scratch.v.h)->value;
 3189|      0|                        const Identifier *fid = alloc->makeIdentifier(fname);
 3190|      0|                        if (f.elements.find(fid) != f.elements.end()) {
  ------------------
  |  Branch (3190:29): [True: 0, False: 0]
  ------------------
 3191|      0|                            throw makeError(ast.location,
 3192|      0|                                            "duplicate field name: \"" + encode_utf8(fname) + "\"");
 3193|      0|                        }
 3194|      0|                        f.elements[fid] = arr->elements[f.elementId];
 3195|      0|                    }
 3196|      0|                    f.elementId++;
 3197|       |
 3198|      0|                    if (f.elementId == arr->elements.size()) {
  ------------------
  |  Branch (3198:25): [True: 0, False: 0]
  ------------------
 3199|      0|                        auto env = capture(ast.freeVariables);
 3200|      0|                        scratch =
 3201|      0|                            makeObject<HeapComprehensionObject>(env, ast.value, ast.id, f.elements);
 3202|      0|                    } else {
 3203|      0|                        f.bindings[ast.id] = arr->elements[f.elementId];
 3204|      0|                        ast_ = ast.field;
 3205|      0|                        goto recurse;
 3206|      0|                    }
 3207|      0|                } break;
 3208|       |
 3209|   908k|                case FRAME_STRING_CONCAT: {
  ------------------
  |  Branch (3209:17): [True: 908k, False: 105M]
  ------------------
 3210|   908k|                    const auto &ast = *static_cast<const Binary *>(f.ast);
 3211|   908k|                    const Value &lhs = stack.top().val;
 3212|   908k|                    UString output;
 3213|   908k|                    if (lhs.t == Value::STRING) {
  ------------------
  |  Branch (3213:25): [True: 740k, False: 168k]
  ------------------
 3214|   740k|                        output.append(static_cast<const HeapString *>(lhs.v.h)->value);
 3215|   740k|                    } else {
 3216|   168k|                        scratch = lhs;
 3217|   168k|                        output.append(toString(ast.left->location));
 3218|   168k|                    }
 3219|   908k|                    const Value &rhs = stack.top().val2;
 3220|   908k|                    if (rhs.t == Value::STRING) {
  ------------------
  |  Branch (3220:25): [True: 807k, False: 101k]
  ------------------
 3221|   807k|                        output.append(static_cast<const HeapString *>(rhs.v.h)->value);
 3222|   807k|                    } else {
 3223|   101k|                        scratch = rhs;
 3224|   101k|                        output.append(toString(ast.right->location));
 3225|   101k|                    }
 3226|   908k|                    scratch = makeString(output);
 3227|   908k|                } break;
 3228|       |
 3229|  1.50M|                case FRAME_UNARY: {
  ------------------
  |  Branch (3229:17): [True: 1.50M, False: 104M]
  ------------------
 3230|  1.50M|                    const auto &ast = *static_cast<const Unary *>(f.ast);
 3231|  1.50M|                    switch (scratch.t) {
 3232|  1.42M|                        case Value::BOOLEAN:
  ------------------
  |  Branch (3232:25): [True: 1.42M, False: 73.9k]
  ------------------
 3233|  1.42M|                            if (ast.op == UOP_NOT) {
  ------------------
  |  Branch (3233:33): [True: 1.42M, False: 1]
  ------------------
 3234|  1.42M|                                scratch = makeBoolean(!scratch.v.b);
 3235|  1.42M|                            } else {
 3236|      1|                                throw makeError(ast.location,
 3237|      1|                                                "unary operator " + uop_string(ast.op) +
 3238|      1|                                                    " does not operate on booleans.");
 3239|      1|                            }
 3240|  1.42M|                            break;
 3241|       |
 3242|  1.42M|                        case Value::NUMBER:
  ------------------
  |  Branch (3242:25): [True: 73.8k, False: 1.42M]
  ------------------
 3243|  73.8k|                            switch (ast.op) {
 3244|  7.02k|                                case UOP_PLUS: break;
  ------------------
  |  Branch (3244:33): [True: 7.02k, False: 66.8k]
  ------------------
 3245|       |
 3246|  65.7k|                                case UOP_MINUS: scratch = makeNumber(-scratch.v.d); break;
  ------------------
  |  Branch (3246:33): [True: 65.7k, False: 8.12k]
  ------------------
 3247|       |
 3248|  1.09k|                                case UOP_BITWISE_NOT:
  ------------------
  |  Branch (3248:33): [True: 1.09k, False: 72.7k]
  ------------------
 3249|  1.09k|                                    scratch = makeNumber(~(long)(scratch.v.d));
 3250|  1.09k|                                    break;
 3251|       |
 3252|     14|                                default:
  ------------------
  |  Branch (3252:33): [True: 14, False: 73.8k]
  ------------------
 3253|     14|                                    throw makeError(ast.location,
 3254|     14|                                                    "unary operator " + uop_string(ast.op) +
 3255|     14|                                                        " does not operate on numbers.");
 3256|  73.8k|                            }
 3257|  73.8k|                            break;
 3258|       |
 3259|  73.8k|                        default:
  ------------------
  |  Branch (3259:25): [True: 84, False: 1.50M]
  ------------------
 3260|     84|                            throw makeError(ast.location,
 3261|     84|                                            "unary operator " + uop_string(ast.op) +
 3262|     84|                                                " does not operate on type " + type_str(scratch));
 3263|  1.50M|                    }
 3264|  1.50M|                } break;
 3265|       |
 3266|  1.50M|                case FRAME_BUILTIN_JOIN_STRINGS: {
  ------------------
  |  Branch (3266:17): [True: 0, False: 106M]
  ------------------
 3267|      0|                    joinString(f.first, f.str, f.val, f.elementId, scratch);
 3268|      0|                    f.elementId++;
 3269|      0|                    auto *ast = joinStrings();
 3270|      0|                    if (ast != nullptr) {
  ------------------
  |  Branch (3270:25): [True: 0, False: 0]
  ------------------
 3271|      0|                        ast_ = ast;
 3272|      0|                        goto recurse;
 3273|      0|                    }
 3274|      0|                } break;
 3275|       |
 3276|      0|                case FRAME_BUILTIN_JOIN_ARRAYS: {
  ------------------
  |  Branch (3276:17): [True: 0, False: 106M]
  ------------------
 3277|      0|                    joinArray(f.first, f.thunks, f.val, f.elementId, scratch);
 3278|      0|                    f.elementId++;
 3279|      0|                    auto *ast = joinArrays();
 3280|      0|                    if (ast != nullptr) {
  ------------------
  |  Branch (3280:25): [True: 0, False: 0]
  ------------------
 3281|      0|                        ast_ = ast;
 3282|      0|                        goto recurse;
 3283|      0|                    }
 3284|      0|                } break;
 3285|       |
 3286|      0|                 case FRAME_BUILTIN_DECODE_UTF8: {
  ------------------
  |  Branch (3286:18): [True: 0, False: 106M]
  ------------------
 3287|      0|                    auto *ast = decodeUTF8();
 3288|      0|                    if (ast != nullptr) {
  ------------------
  |  Branch (3288:25): [True: 0, False: 0]
  ------------------
 3289|      0|                        ast_ = ast;
 3290|      0|                        goto recurse;
 3291|      0|                    }
 3292|      0|                } break;
 3293|       |
 3294|      0|                default:
  ------------------
  |  Branch (3294:17): [True: 0, False: 106M]
  ------------------
 3295|      0|                    std::cerr << "INTERNAL ERROR: Unknown FrameKind:  " << f.kind << std::endl;
 3296|      0|                    std::abort();
 3297|   106M|            }
 3298|       |
 3299|  46.2M|        popframe:;
 3300|       |
 3301|  46.2M|            stack.pop();
 3302|       |
 3303|  54.9M|        replaceframe:;
 3304|  54.9M|        }
 3305|  53.7M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack8newFrameIJNS1_9FrameKindEPKNS0_3ASTEEEEvDpT_:
  388|  42.0M|    {
  389|  42.0M|        stack.emplace_back(args...);
  390|  42.0M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15FrameC2ERKNS1_9FrameKindEPKNS0_3ASTE:
  194|  42.0M|        : kind(kind),
  195|  42.0M|          ast(ast),
  196|  42.0M|          location(ast->location),
  197|  42.0M|          tailCall(false),
  198|  42.0M|          elementId(0),
  199|  42.0M|          first(false),
  200|  42.0M|          context(NULL),
  201|  42.0M|          self(NULL),
  202|  42.0M|          offset(0)
  203|  42.0M|    {
  204|  42.0M|        val.t = Value::NULL_TYPE;
  205|  42.0M|        val2.t = Value::NULL_TYPE;
  206|  42.0M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack14getSelfBindingERPNS1_10HeapObjectERj:
  444|  24.7M|    {
  445|  24.7M|        self = nullptr;
  446|  24.7M|        offset = 0;
  447|  80.9M|        for (int i = stack.size() - 1; i >= 0; --i) {
  ------------------
  |  Branch (447:40): [True: 80.9M, False: 17.3k]
  ------------------
  448|  80.9M|            if (stack[i].isCall()) {
  ------------------
  |  Branch (448:17): [True: 24.7M, False: 56.1M]
  ------------------
  449|  24.7M|                self = stack[i].self;
  450|  24.7M|                offset = stack[i].offset;
  451|  24.7M|                return;
  452|  24.7M|            }
  453|  80.9M|        }
  454|  24.7M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJRPKNS0_10IdentifierERPNS1_10HeapObjectERjRKPNS0_3ASTEEEEPT_DpOT0_:
  577|  16.4M|    {
  578|  16.4M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  16.4M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 16.2k, False: 16.4M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  16.2k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  16.2k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  16.2k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  16.2k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 16.2k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|  2.61M|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 2.61M, False: 16.2k]
  ------------------
  597|  2.61M|                heap.markFrom(sourceVal.second);
  598|  2.61M|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  16.2k|            heap.sweep();
  602|  16.2k|        }
  603|  16.4M|        return r;
  604|  16.4M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter7captureERKNSt3__16vectorIPKNS0_10IdentifierENS3_9allocatorIS7_EEEE:
  861|  34.6M|    {
  862|  34.6M|        BindingFrame env;
  863|   853M|        for (auto fv : free_vars) {
  ------------------
  |  Branch (863:22): [True: 853M, False: 34.6M]
  ------------------
  864|   853M|            auto *th = stack.lookUpVar(fv);
  865|   853M|            env[fv] = th;
  866|   853M|        }
  867|  34.6M|        return env;
  868|  34.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack8newFrameIJNS1_9FrameKindENS0_13LocationRangeEEEEvDpT_:
  388|  8.89M|    {
  389|  8.89M|        stack.emplace_back(args...);
  390|  8.89M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter11makeClosureERKNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS3_4lessIS7_EENS3_9allocatorINS3_4pairIKS7_S9_EEEEEEPNS1_10HeapObjectEjRKNS3_6vectorINS1_11HeapClosure5ParamENSC_ISO_EEEEPNS0_3ASTE:
  650|  1.60M|    {
  651|  1.60M|        Value r;
  652|  1.60M|        r.t = Value::FUNCTION;
  653|  1.60M|        r.v.h = makeHeap<HeapClosure>(env, self, offset, params, body, "");
  654|  1.60M|        return r;
  655|  1.60M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_11HeapClosureEJRKNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEERPNS1_10HeapObjectERjRKNS5_6vectorINS4_5ParamENSE_ISR_EEEERPNS0_3ASTERA1_KcEEEPT_DpOT0_:
  577|  1.60M|    {
  578|  1.60M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  1.60M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 3.05k, False: 1.60M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  3.05k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  3.05k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  3.05k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  3.05k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 3.05k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|   491k|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 491k, False: 3.05k]
  ------------------
  597|   491k|                heap.markFrom(sourceVal.second);
  598|   491k|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  3.05k|            heap.sweep();
  602|  3.05k|        }
  603|  1.60M|        return r;
  604|  1.60M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter6importERKNS0_13LocationRangeEPKNS0_13LiteralStringE:
  792|     23|    {
  793|     23|        ImportCacheValue *input = importData(loc, file);
  794|     23|        if (input->thunk == nullptr) {
  ------------------
  |  Branch (794:13): [True: 0, False: 23]
  ------------------
  795|      0|            Tokens tokens = jsonnet_lex(input->foundHere, input->content.c_str());
  796|      0|            AST *expr = jsonnet_parse(alloc, tokens);
  797|      0|            jsonnet_desugar(alloc, expr, nullptr);
  798|      0|            jsonnet_static_analysis(expr);
  799|       |            // If no errors then populate cache.
  800|      0|            auto *thunk = makeHeap<HeapThunk>(idImport, nullptr, 0, expr);
  801|      0|            input->thunk = thunk;
  802|      0|        }
  803|     23|        return input->thunk;
  804|     23|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10importDataERKNS0_13LocationRangeEPKNS0_13LiteralStringE:
  816|     34|    {
  817|       |        // `dir` is passed to the importCallback, which may be externally defined.
  818|       |        // For backwards compatibility, we need to keep the trailing directory separator.
  819|       |        // For example, the default callback in libjsonnet.cpp joins paths with simple
  820|       |        // string concatenation. Other (external) implementations might do the same.
  821|     34|        std::string dir = path_dir_with_trailing_separator(loc.file);
  822|       |
  823|     34|        const UString &path = file->value;
  824|       |
  825|     34|        std::pair<std::string, UString> key(dir, path);
  826|     34|        ImportCacheValue *cached_value = cachedImports[key];
  827|     34|        if (cached_value != nullptr)
  ------------------
  |  Branch (827:13): [True: 0, False: 34]
  ------------------
  828|      0|            return cached_value;
  829|       |
  830|     34|        char *found_here_cptr;
  831|     34|        char *buf = NULL;
  832|     34|        size_t buflen = 0;
  833|     34|        int result = importCallback(importCallbackContext,
  834|     34|                                    dir.c_str(),
  835|     34|                                    encode_utf8(path).c_str(),
  836|     34|                                    &found_here_cptr,
  837|     34|                                    &buf,
  838|     34|                                    &buflen);
  839|       |
  840|     34|        std::string input(buf, buflen);
  841|     34|        ::free(buf);
  842|       |
  843|     34|        if (result == 1) {  // failure
  ------------------
  |  Branch (843:13): [True: 34, False: 0]
  ------------------
  844|     34|            std::string epath = encode_utf8(jsonnet_string_escape(path, false));
  845|     34|            std::string msg = "couldn't open import \"" + epath + "\": ";
  846|     34|            msg += input;
  847|     34|            throw makeError(loc, msg);
  848|     34|        }
  849|       |
  850|      0|        auto *input_ptr = new ImportCacheValue();
  851|      0|        input_ptr->foundHere = found_here_cptr;
  852|      0|        input_ptr->content = input;
  853|      0|        input_ptr->thunk = nullptr;  // May be filled in later by import().
  854|      0|        ::free(found_here_cptr);
  855|      0|        cachedImports[key] = input_ptr;
  856|      0|        return input_ptr;
  857|     34|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJRKPKNS0_10IdentifierERPNS1_10HeapObjectERjRKPNS0_3ASTEEEEPT_DpOT0_:
  577|  14.6M|    {
  578|  14.6M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  14.6M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 16.6k, False: 14.5M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  16.6k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  16.6k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  16.6k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  16.6k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 16.6k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|  2.68M|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 2.68M, False: 16.6k]
  ------------------
  597|  2.68M|                heap.markFrom(sourceVal.second);
  598|  2.68M|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  16.6k|            heap.sweep();
  602|  16.6k|        }
  603|  14.6M|        return r;
  604|  14.6M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10makeObjectINS1_16HeapSimpleObjectEJNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEENS6_IS9_NS4_5FieldESD_NSE_INSF_ISG_SK_EEEEEENS5_4listIPNS0_3ASTENSE_ISQ_EEEEEEENS1_5ValueEDpT0_:
  671|  1.43M|    {
  672|  1.43M|        Value r;
  673|  1.43M|        r.t = Value::OBJECT;
  674|  1.43M|        r.v.h = makeHeap<T>(args...);
  675|  1.43M|        return r;
  676|  1.43M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_16HeapSimpleObjectEJRNSt3__13mapIPKNS0_10IdentifierEPNS1_9HeapThunkENS5_4lessIS9_EENS5_9allocatorINS5_4pairIKS9_SB_EEEEEERNS6_IS9_NS4_5FieldESD_NSE_INSF_ISG_SL_EEEEEERNS5_4listIPNS0_3ASTENSE_ISS_EEEEEEEPT_DpOT0_:
  577|  1.43M|    {
  578|  1.43M|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  1.43M|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 1.69k, False: 1.43M]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  1.69k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  1.69k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  1.69k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  1.69k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 1.69k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|   272k|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 272k, False: 1.69k]
  ------------------
  597|   272k|                heap.markFrom(sourceVal.second);
  598|   272k|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  1.69k|            heap.sweep();
  602|  1.69k|        }
  603|  1.43M|        return r;
  604|  1.43M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack9lookUpVarEPKNS0_10IdentifierE:
  269|   874M|    {
  270|  1.26G|        for (int i = stack.size() - 1; i >= 0; --i) {
  ------------------
  |  Branch (270:40): [True: 1.26G, False: 0]
  ------------------
  271|  1.26G|            const auto &binds = stack[i].bindings;
  272|  1.26G|            auto it = binds.find(id);
  273|  1.26G|            if (it != binds.end()) {
  ------------------
  |  Branch (273:17): [True: 874M, False: 391M]
  ------------------
  274|   874M|                return it->second;
  275|   874M|            }
  276|   391M|            if (stack[i].isCall())
  ------------------
  |  Branch (276:17): [True: 0, False: 391M]
  ------------------
  277|      0|                break;
  278|   391M|        }
  279|      0|        return nullptr;
  280|   874M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack4sizeEv:
  263|   119M|    {
  264|   119M|        return stack.size();
  265|   119M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJRPKNS0_10IdentifierERPNS1_10HeapObjectERjRKPKNS0_3ASTEEEEPT_DpOT0_:
  577|     29|    {
  578|     29|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|     29|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 0, False: 29]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|      0|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|      0|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|      0|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|      0|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 0]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|      0|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 0, False: 0]
  ------------------
  597|      0|                heap.markFrom(sourceVal.second);
  598|      0|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|      0|            heap.sweep();
  602|      0|        }
  603|     29|        return r;
  604|     29|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10findObjectEPKNS0_10IdentifierEPNS1_10HeapObjectEjRj:
  698|   528M|    {
  699|   528M|        if (auto *ext = dynamic_cast<HeapExtendedObject *>(curr)) {
  ------------------
  |  Branch (699:19): [True: 260M, False: 268M]
  ------------------
  700|   260M|            auto *r = findObject(f, ext->right, start_from, counter);
  701|   260M|            if (r)
  ------------------
  |  Branch (701:17): [True: 3.15M, False: 256M]
  ------------------
  702|  3.15M|                return r;
  703|   256M|            auto *l = findObject(f, ext->left, start_from, counter);
  704|   256M|            if (l)
  ------------------
  |  Branch (704:17): [True: 186M, False: 70.9M]
  ------------------
  705|   186M|                return l;
  706|   268M|        } else if (auto *ext = dynamic_cast<HeapRestrictedObject *>(curr)) {
  ------------------
  |  Branch (706:26): [True: 0, False: 268M]
  ------------------
  707|       |            // The RestrictedObject is on the 'right', that is it is checked/counted first.
  708|      0|            if (counter >= start_from) {
  ------------------
  |  Branch (708:17): [True: 0, False: 0]
  ------------------
  709|      0|                if (ext->retainedKeys.find(f) == ext->retainedKeys.end()) {
  ------------------
  |  Branch (709:21): [True: 0, False: 0]
  ------------------
  710|      0|                    counter += 1 + countLeaves(ext->obj);
  711|      0|                    return nullptr;
  712|      0|                }
  713|      0|            }
  714|      0|            ++counter;
  715|      0|            return findObject(f, ext->obj, start_from, counter);
  716|   268M|        } else {
  717|   268M|            if (counter >= start_from) {
  ------------------
  |  Branch (717:17): [True: 31.8M, False: 236M]
  ------------------
  718|  31.8M|                if (auto *simp = dynamic_cast<HeapSimpleObject *>(curr)) {
  ------------------
  |  Branch (718:27): [True: 31.8M, False: 0]
  ------------------
  719|  31.8M|                    auto it = simp->fields.find(f);
  720|  31.8M|                    if (it != simp->fields.end()) {
  ------------------
  |  Branch (720:25): [True: 11.2M, False: 20.5M]
  ------------------
  721|  11.2M|                        return simp;
  722|  11.2M|                    }
  723|  31.8M|                } else if (auto *comp = dynamic_cast<HeapComprehensionObject *>(curr)) {
  ------------------
  |  Branch (723:34): [True: 0, False: 0]
  ------------------
  724|      0|                    auto it = comp->compValues.find(f);
  725|      0|                    if (it != comp->compValues.end()) {
  ------------------
  |  Branch (725:25): [True: 0, False: 0]
  ------------------
  726|      0|                        return comp;
  727|      0|                    }
  728|      0|                }
  729|  31.8M|            }
  730|   257M|            counter++;
  731|   257M|        }
  732|   328M|        return nullptr;
  733|   528M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter13callSourceValEPKNS0_3ASTEPNS1_9HeapThunkENSt3__16vectorIS7_NS8_9allocatorIS7_EEEE:
 2070|  10.2k|    const AST *callSourceVal(const AST *ast, HeapThunk *sourceVal, std::vector<HeapThunk*> args) {
 2071|  10.2k|        assert(sourceVal != nullptr);
  ------------------
  |  Branch (2071:9): [True: 10.2k, False: 0]
  ------------------
 2072|  10.2k|        assert(sourceVal->filled);
  ------------------
  |  Branch (2072:9): [True: 10.2k, False: 0]
  ------------------
 2073|  10.2k|        assert(sourceVal->content.t == Value::FUNCTION);
  ------------------
  |  Branch (2073:9): [True: 10.2k, False: 0]
  ------------------
 2074|  10.2k|        auto *func = static_cast<HeapClosure *>(sourceVal->content.v.h);
 2075|  10.2k|        BindingFrame up_values = func->upValues;
 2076|  30.6k|        for (size_t i = 0; i < args.size(); ++i) {
  ------------------
  |  Branch (2076:28): [True: 20.4k, False: 10.2k]
  ------------------
 2077|  20.4k|            up_values.insert({func->params[i].id, args[i]});
 2078|  20.4k|        }
 2079|  10.2k|        stack.newCall(ast->location, func, func->self, func->offset, up_values);
 2080|  10.2k|        return func->body;
 2081|  10.2k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter17safeDoubleToInt64EdRKNS0_13LocationRangeE:
  919|  15.1k|    int64_t safeDoubleToInt64(double value, const internal::LocationRange& loc) {
  920|  15.1k|        if (std::isnan(value) || std::isinf(value)) {
  ------------------
  |  Branch (920:13): [True: 0, False: 15.1k]
  |  Branch (920:34): [True: 0, False: 15.1k]
  ------------------
  921|      0|            throw internal::StaticError(loc, "numeric value is not finite");
  922|      0|        }
  923|       |
  924|       |        // Constants for safe double-to-int conversion
  925|       |        // Jsonnet uses IEEE 754 doubles, which precisely represent integers in the range [-2^53 + 1, 2^53 - 1].
  926|  15.1k|        constexpr int64_t DOUBLE_MAX_SAFE_INTEGER = (1LL << 53) - 1;
  927|  15.1k|        constexpr int64_t DOUBLE_MIN_SAFE_INTEGER = -((1LL << 53) - 1);
  928|       |
  929|       |        // Check if the value is within the safe integer range
  930|  15.1k|        if (value < DOUBLE_MIN_SAFE_INTEGER || value > DOUBLE_MAX_SAFE_INTEGER) {
  ------------------
  |  Branch (930:13): [True: 2, False: 15.1k]
  |  Branch (930:48): [True: 5, False: 15.1k]
  ------------------
  931|      7|            throw makeError(loc, "numeric value outside safe integer range for bitwise operation.");
  932|      7|        }
  933|  15.1k|        return static_cast<int64_t>(value);
  934|  15.1k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter10makeObjectINS1_18HeapExtendedObjectEJPNS1_10HeapObjectES6_EEENS1_5ValueEDpT0_:
  671|   881k|    {
  672|   881k|        Value r;
  673|   881k|        r.t = Value::OBJECT;
  674|   881k|        r.v.h = makeHeap<T>(args...);
  675|   881k|        return r;
  676|   881k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_18HeapExtendedObjectEJRPNS1_10HeapObjectES7_EEEPT_DpOT0_:
  577|   881k|    {
  578|   881k|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|   881k|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 1.10k, False: 880k]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|  1.10k|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|  1.10k|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|  1.10k|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|  1.10k|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 1.10k]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|   178k|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 178k, False: 1.10k]
  ------------------
  597|   178k|                heap.markFrom(sourceVal.second);
  598|   178k|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|  1.10k|            heap.sweep();
  602|  1.10k|        }
  603|   881k|        return r;
  604|   881k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8toStringERKNS0_13LocationRangeE:
 1979|   264k|    {
 1980|   264k|        return manifestJson(loc, false, U"");
 1981|   264k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter11countLeavesEPNS1_10HeapObjectE:
  876|  1.16G|    {
  877|  1.16G|        if (auto *ext = dynamic_cast<HeapExtendedObject *>(obj)) {
  ------------------
  |  Branch (877:19): [True: 580M, False: 582M]
  ------------------
  878|   580M|            return countLeaves(ext->left) + countLeaves(ext->right);
  879|   582M|        } else if (auto *ext = dynamic_cast<HeapRestrictedObject *>(obj)) {
  ------------------
  |  Branch (879:26): [True: 0, False: 582M]
  ------------------
  880|      0|            return countLeaves(ext->obj) + 1;
  881|   582M|        } else {
  882|       |            // Must be a HeapLeafObject.
  883|   582M|            return 1;
  884|   582M|        }
  885|  1.16G|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter11objectIndexERKNS0_13LocationRangeEPNS1_10HeapObjectEPKNS0_10IdentifierEj:
 2018|  10.1M|    {
 2019|  10.1M|        unsigned found_at = 0;
 2020|  10.1M|        HeapObject *self = obj;
 2021|  10.1M|        HeapLeafObject *found = findObject(f, obj, offset, found_at);
 2022|  10.1M|        if (found == nullptr) {
  ------------------
  |  Branch (2022:13): [True: 32, False: 10.1M]
  ------------------
 2023|     32|            throw makeError(loc, "field does not exist: " + encode_utf8(f->name));
 2024|     32|        }
 2025|  10.1M|        if (auto *simp = dynamic_cast<HeapSimpleObject *>(found)) {
  ------------------
  |  Branch (2025:19): [True: 10.1M, False: 0]
  ------------------
 2026|  10.1M|            auto it = simp->fields.find(f);
 2027|  10.1M|            const AST *body = it->second.body;
 2028|       |
 2029|  10.1M|            stack.newCall(loc, simp, self, found_at, simp->upValues);
 2030|  10.1M|            return body;
 2031|  10.1M|        } else {
 2032|       |            // If a HeapLeafObject is not HeapSimpleObject, it must be HeapComprehensionObject.
 2033|      0|            auto *comp = static_cast<HeapComprehensionObject *>(found);
 2034|      0|            auto it = comp->compValues.find(f);
 2035|      0|            auto *th = it->second;
 2036|      0|            BindingFrame binds = comp->upValues;
 2037|      0|            binds[comp->id] = th;
 2038|      0|            stack.newCall(loc, comp, self, found_at, binds);
 2039|      0|            return comp->value;
 2040|      0|        }
 2041|  10.1M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_15Stack26alreadyExecutingInvariantsEPNS1_10HeapObjectE:
  458|  8.90M|    {
  459|  2.59G|        for (int i = stack.size() - 1; i >= 0; --i) {
  ------------------
  |  Branch (459:40): [True: 2.58G, False: 8.89M]
  ------------------
  460|  2.58G|            if (stack[i].kind == FRAME_INVARIANTS) {
  ------------------
  |  Branch (460:17): [True: 1.83M, False: 2.58G]
  ------------------
  461|  1.83M|                if (stack[i].self == self)
  ------------------
  |  Branch (461:21): [True: 7.11k, False: 1.82M]
  ------------------
  462|  7.11k|                    return true;
  463|  1.83M|            }
  464|  2.58G|        }
  465|  8.89M|        return false;
  466|  8.90M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter16objectInvariantsEPNS1_10HeapObjectES4_RjRNSt3__16vectorIPNS1_9HeapThunkENS6_9allocatorIS9_EEEE:
 1992|  33.1M|    {
 1993|  33.1M|        if (auto *ext = dynamic_cast<HeapExtendedObject *>(curr)) {
  ------------------
  |  Branch (1993:19): [True: 12.1M, False: 21.0M]
  ------------------
 1994|  12.1M|            objectInvariants(ext->right, self, counter, thunks);
 1995|  12.1M|            objectInvariants(ext->left, self, counter, thunks);
 1996|  21.0M|        } else if (auto *ext = dynamic_cast<HeapRestrictedObject *>(curr)) {
  ------------------
  |  Branch (1996:26): [True: 0, False: 21.0M]
  ------------------
 1997|      0|            objectInvariants(ext->obj, self, counter, thunks);
 1998|  21.0M|        } else {
 1999|  21.0M|            if (auto *simp = dynamic_cast<HeapSimpleObject *>(curr)) {
  ------------------
  |  Branch (1999:23): [True: 21.0M, False: 0]
  ------------------
 2000|  21.0M|                for (AST *assert : simp->asserts) {
  ------------------
  |  Branch (2000:34): [True: 81.0k, False: 21.0M]
  ------------------
 2001|  81.0k|                    auto *el_th = makeHeap<HeapThunk>(idInvariant, self, counter, assert);
 2002|  81.0k|                    el_th->upValues = simp->upValues;
 2003|  81.0k|                    thunks.push_back(el_th);
 2004|  81.0k|                }
 2005|  21.0M|            }
 2006|  21.0M|            counter++;
 2007|  21.0M|        }
 2008|  33.1M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter8makeHeapINS1_9HeapThunkEJRPKNS0_10IdentifierERPNS1_10HeapObjectERjRPNS0_3ASTEEEEPT_DpOT0_:
  577|  81.0k|    {
  578|  81.0k|        T *r = heap.makeEntity<T, Args...>(std::forward<Args>(args)...);
  579|  81.0k|        if (heap.checkHeap()) {  // Do a GC cycle?
  ------------------
  |  Branch (579:13): [True: 170, False: 80.9k]
  ------------------
  580|       |            // Avoid the object we just made being collected.
  581|    170|            heap.markFrom(r);
  582|       |
  583|       |            // Mark from the stack.
  584|    170|            stack.mark(heap);
  585|       |
  586|       |            // Mark from the scratch register
  587|    170|            heap.markFrom(scratch);
  588|       |
  589|       |            // Mark from cached imports
  590|    170|            for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (590:35): [True: 0, False: 170]
  ------------------
  591|      0|                HeapThunk *thunk = pair.second->thunk;
  592|      0|                if (thunk != nullptr)
  ------------------
  |  Branch (592:21): [True: 0, False: 0]
  ------------------
  593|      0|                    heap.markFrom(thunk);
  594|      0|            }
  595|       |
  596|  27.3k|            for (const auto &sourceVal : sourceVals) {
  ------------------
  |  Branch (596:40): [True: 27.3k, False: 170]
  ------------------
  597|  27.3k|                heap.markFrom(sourceVal.second);
  598|  27.3k|            }
  599|       |
  600|       |            // Delete unreachable objects.
  601|    170|            heap.sweep();
  602|    170|        }
  603|  81.0k|        return r;
  604|  81.0k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter12manifestJsonERKNS0_13LocationRangeEbRKNSt3__112basic_stringIDiNS6_11char_traitsIDiEENS6_9allocatorIDiEEEE:
 3315|  2.60M|    {
 3316|       |        // Printing fields means evaluating and binding them, which can trigger
 3317|       |        // garbage collection.
 3318|       |
 3319|  2.60M|        UStringStream ss;
 3320|  2.60M|        switch (scratch.t) {
  ------------------
  |  Branch (3320:17): [True: 2.60M, False: 0]
  ------------------
 3321|   174k|            case Value::ARRAY: {
  ------------------
  |  Branch (3321:13): [True: 174k, False: 2.42M]
  ------------------
 3322|   174k|                HeapArray *arr = static_cast<HeapArray *>(scratch.v.h);
 3323|   174k|                if (arr->elements.size() == 0) {
  ------------------
  |  Branch (3323:21): [True: 43.3k, False: 131k]
  ------------------
 3324|  43.3k|                    ss << U"[ ]";
 3325|   131k|                } else {
 3326|   131k|                    const char32_t *prefix = multiline ? U"[\n" : U"[";
  ------------------
  |  Branch (3326:46): [True: 22.8k, False: 108k]
  ------------------
 3327|   131k|                    UString indent2 = multiline ? indent + U"   " : indent;
  ------------------
  |  Branch (3327:39): [True: 22.8k, False: 108k]
  ------------------
 3328|  1.27M|                    for (auto *thunk : arr->elements) {
  ------------------
  |  Branch (3328:38): [True: 1.27M, False: 131k]
  ------------------
 3329|  1.27M|                        LocationRange tloc = thunk->body == nullptr ? loc : thunk->body->location;
  ------------------
  |  Branch (3329:46): [True: 0, False: 1.27M]
  ------------------
 3330|  1.27M|                        if (thunk->filled) {
  ------------------
  |  Branch (3330:29): [True: 0, False: 1.27M]
  ------------------
 3331|      0|                            stack.newCall(loc, thunk, nullptr, 0, BindingFrame{});
 3332|       |                            // Keep arr alive when scratch is overwritten
 3333|      0|                            stack.top().val = scratch;
 3334|      0|                            scratch = thunk->content;
 3335|  1.27M|                        } else {
 3336|  1.27M|                            stack.newCall(loc, thunk, thunk->self, thunk->offset, thunk->upValues);
 3337|       |                            // Keep arr alive when scratch is overwritten
 3338|  1.27M|                            stack.top().val = scratch;
 3339|  1.27M|                            evaluate(thunk->body, stack.size());
 3340|  1.27M|                        }
 3341|  1.27M|                        auto element = manifestJson(tloc, multiline, indent2);
 3342|       |                        // Restore scratch
 3343|  1.27M|                        scratch = stack.top().val;
 3344|  1.27M|                        stack.pop();
 3345|  1.27M|                        ss << prefix << indent2 << element;
 3346|  1.27M|                        prefix = multiline ? U",\n" : U", ";
  ------------------
  |  Branch (3346:34): [True: 71.5k, False: 1.20M]
  ------------------
 3347|  1.27M|                    }
 3348|   131k|                    ss << (multiline ? U"\n" : U"") << indent << U"]";
  ------------------
  |  Branch (3348:28): [True: 22.4k, False: 108k]
  ------------------
 3349|   131k|                }
 3350|   174k|            } break;
 3351|       |
 3352|  57.2k|            case Value::BOOLEAN: ss << (scratch.v.b ? U"true" : U"false"); break;
  ------------------
  |  Branch (3352:13): [True: 57.2k, False: 2.54M]
  |  Branch (3352:41): [True: 23.0k, False: 34.2k]
  ------------------
 3353|       |
 3354|  1.73M|            case Value::NUMBER: ss << decode_utf8(jsonnet_unparse_number(scratch.v.d)); break;
  ------------------
  |  Branch (3354:13): [True: 1.73M, False: 861k]
  ------------------
 3355|       |
 3356|      2|            case Value::FUNCTION:
  ------------------
  |  Branch (3356:13): [True: 2, False: 2.60M]
  ------------------
 3357|      2|                throw makeError(loc, "couldn't manifest function in JSON output.");
 3358|       |
 3359|    584|            case Value::NULL_TYPE: ss << U"null"; break;
  ------------------
  |  Branch (3359:13): [True: 584, False: 2.59M]
  ------------------
 3360|       |
 3361|   547k|            case Value::OBJECT: {
  ------------------
  |  Branch (3361:13): [True: 547k, False: 2.05M]
  ------------------
 3362|   547k|                auto *obj = static_cast<HeapObject *>(scratch.v.h);
 3363|   547k|                runInvariants(loc, obj);
 3364|       |                // Using std::map has the useful side-effect of ordering the fields
 3365|       |                // alphabetically.
 3366|   547k|                std::map<UString, const Identifier *> fields;
 3367|   670k|                for (const auto &f : objectFields(obj, true)) {
  ------------------
  |  Branch (3367:36): [True: 670k, False: 547k]
  ------------------
 3368|   670k|                    fields[f->name] = f;
 3369|   670k|                }
 3370|   547k|                if (fields.size() == 0) {
  ------------------
  |  Branch (3370:21): [True: 174k, False: 373k]
  ------------------
 3371|   174k|                    ss << U"{ }";
 3372|   373k|                } else {
 3373|   373k|                    UString indent2 = multiline ? indent + U"   " : indent;
  ------------------
  |  Branch (3373:39): [True: 12.8k, False: 360k]
  ------------------
 3374|   373k|                    const char32_t *prefix = multiline ? U"{\n" : U"{";
  ------------------
  |  Branch (3374:46): [True: 12.8k, False: 360k]
  ------------------
 3375|   597k|                    for (const auto &f : fields) {
  ------------------
  |  Branch (3375:40): [True: 597k, False: 373k]
  ------------------
 3376|       |                        // pushes FRAME_CALL
 3377|   597k|                        const AST *body = objectIndex(loc, obj, f.second, 0);
 3378|   597k|                        stack.top().val = scratch;
 3379|   597k|                        evaluate(body, stack.size());
 3380|   597k|                        auto vstr = manifestJson(body->location, multiline, indent2);
 3381|       |                        // Reset scratch so that the object we're manifesting doesn't
 3382|       |                        // get GC'd.
 3383|   597k|                        scratch = stack.top().val;
 3384|   597k|                        stack.pop();
 3385|   597k|                        ss << prefix << indent2 << jsonnet_string_unparse(f.first, false) << U": "
 3386|   597k|                           << vstr;
 3387|   597k|                        prefix = multiline ? U",\n" : U", ";
  ------------------
  |  Branch (3387:34): [True: 46.8k, False: 551k]
  ------------------
 3388|   597k|                    }
 3389|   373k|                    ss << (multiline ? U"\n" : U"") << indent << U"}";
  ------------------
  |  Branch (3389:28): [True: 2.82k, False: 370k]
  ------------------
 3390|   373k|                }
 3391|   547k|            } break;
 3392|       |
 3393|  80.6k|            case Value::STRING: {
  ------------------
  |  Branch (3393:13): [True: 80.6k, False: 2.51M]
  ------------------
 3394|  80.6k|                const UString &str = static_cast<HeapString *>(scratch.v.h)->value;
 3395|  80.6k|                ss << jsonnet_string_unparse(str, false);
 3396|  80.6k|            } break;
 3397|  2.60M|        }
 3398|  2.43M|        return ss.str();
 3399|  2.60M|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter13runInvariantsERKNS0_13LocationRangeEPNS1_10HeapObjectE:
 2044|   547k|    {
 2045|   547k|        if (stack.alreadyExecutingInvariants(self))
  ------------------
  |  Branch (2045:13): [True: 610, False: 547k]
  ------------------
 2046|    610|            return;
 2047|       |
 2048|   547k|        unsigned counter = 0;
 2049|   547k|        unsigned initial_stack_size = stack.size();
 2050|   547k|        stack.newFrame(FRAME_INVARIANTS, loc);
 2051|   547k|        std::vector<HeapThunk *> &thunks = stack.top().thunks;
 2052|   547k|        objectInvariants(self, self, counter, thunks);
 2053|   547k|        if (thunks.size() == 0) {
  ------------------
  |  Branch (2053:13): [True: 533k, False: 13.2k]
  ------------------
 2054|   533k|            stack.pop();
 2055|   533k|            return;
 2056|   533k|        }
 2057|  13.2k|        HeapThunk *thunk = thunks[0];
 2058|  13.2k|        stack.top().elementId = 1;
 2059|  13.2k|        stack.top().self = self;
 2060|  13.2k|        stack.newCall(loc, thunk, thunk->self, thunk->offset, thunk->upValues);
 2061|  13.2k|        evaluate(thunk->body, initial_stack_size);
 2062|  13.2k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111InterpreterD2Ev:
 1024|  3.46k|    {
 1025|  3.46k|        for (const auto &pair : cachedImports) {
  ------------------
  |  Branch (1025:31): [True: 34, False: 3.46k]
  ------------------
 1026|     34|            delete pair.second;
 1027|     34|        }
 1028|  3.46k|    }
vm.cpp:_ZN7jsonnet8internal12_GLOBAL__N_111Interpreter14manifestStreamEb:
 3445|  1.73k|    {
 3446|  1.73k|        std::vector<std::string> r;
 3447|  1.73k|        LocationRange loc("During manifestation");
 3448|  1.73k|        if (scratch.t != Value::ARRAY) {
  ------------------
  |  Branch (3448:13): [True: 1.13k, False: 607]
  ------------------
 3449|  1.13k|            std::stringstream ss;
 3450|  1.13k|            ss << "stream mode: top-level object was a " << type_str(scratch.t) << ", "
 3451|  1.13k|               << "should be an array whose elements hold "
 3452|  1.13k|               << "the JSON for each document in the stream.";
 3453|  1.13k|            throw makeError(loc, ss.str());
 3454|  1.13k|        }
 3455|    607|        auto *arr = static_cast<HeapArray *>(scratch.v.h);
 3456|   474k|        for (auto *thunk : arr->elements) {
  ------------------
  |  Branch (3456:26): [True: 474k, False: 607]
  ------------------
 3457|   474k|            LocationRange tloc = thunk->body == nullptr ? loc : thunk->body->location;
  ------------------
  |  Branch (3457:34): [True: 0, False: 474k]
  ------------------
 3458|   474k|            if (thunk->filled) {
  ------------------
  |  Branch (3458:17): [True: 0, False: 474k]
  ------------------
 3459|      0|                stack.newCall(loc, thunk, nullptr, 0, BindingFrame{});
 3460|       |                // Keep arr alive when scratch is overwritten
 3461|      0|                stack.top().val = scratch;
 3462|      0|                scratch = thunk->content;
 3463|   474k|            } else {
 3464|   474k|                stack.newCall(loc, thunk, thunk->self, thunk->offset, thunk->upValues);
 3465|       |                // Keep arr alive when scratch is overwritten
 3466|   474k|                stack.top().val = scratch;
 3467|   474k|                evaluate(thunk->body, stack.size());
 3468|   474k|            }
 3469|   474k|            UString element = string ? manifestString(tloc) : manifestJson(tloc, true, U"");
  ------------------
  |  Branch (3469:31): [True: 0, False: 474k]
  ------------------
 3470|   474k|            scratch = stack.top().val;
 3471|   474k|            stack.pop();
 3472|   474k|            r.push_back(encode_utf8(element));
 3473|   474k|        }
 3474|    607|        return r;
 3475|  1.73k|    }

_ZN7jsonnet8internal10TraceFrameC2ERKNS0_13LocationRangeERKNSt3__112basic_stringIcNS5_11char_traitsIcEENS5_9allocatorIcEEEE:
   33|   235k|        : location(location), name(name)
   34|   235k|    {
   35|   235k|    }
_ZN7jsonnet8internal12RuntimeErrorC2ENSt3__16vectorINS0_10TraceFrameENS2_9allocatorIS4_EEEERKNS2_12basic_stringIcNS2_11char_traitsIcEENS5_IcEEEE:
   45|  2.95k|        : stackTrace(stack_trace), msg(msg)
   46|  2.95k|    {
   47|  2.95k|    }

_ZN2c43yml9CallbacksC2Ev:
31979|      2|    m_user_data(nullptr),
31980|       |    #ifndef RYML_NO_DEFAULT_CALLBACKS
31981|      2|    m_allocate(allocate_impl),
31982|      2|    m_free(free_impl),
31983|      2|    m_error(error_impl)
31984|       |    #else
31985|       |    m_allocate(nullptr),
31986|       |    m_free(nullptr),
31987|       |    m_error(nullptr)
31988|       |    #endif
31989|      2|{
31990|      2|}

