aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/yuescript/yue_compiler.cpp95
-rw-r--r--src/yuescript/yue_parser.cpp2
-rw-r--r--src/yuescript/yuescript.h8
3 files changed, 59 insertions, 46 deletions
diff --git a/src/yuescript/yue_compiler.cpp b/src/yuescript/yue_compiler.cpp
index e7fe0c0..a2a1864 100644
--- a/src/yuescript/yue_compiler.cpp
+++ b/src/yuescript/yue_compiler.cpp
@@ -75,7 +75,7 @@ static std::unordered_set<std::string> Metamethods = {
75 "close"s // Lua 5.4 75 "close"s // Lua 5.4
76}; 76};
77 77
78const std::string_view version = "0.27.0"sv; 78const std::string_view version = "0.27.3"sv;
79const std::string_view extension = "yue"sv; 79const std::string_view extension = "yue"sv;
80 80
81class CompileError : public std::logic_error { 81class CompileError : public std::logic_error {
@@ -437,7 +437,6 @@ private:
437#endif 437#endif
438 std::unique_ptr<std::unordered_map<std::string, VarType>> vars; 438 std::unique_ptr<std::unordered_map<std::string, VarType>> vars;
439 std::unique_ptr<std::unordered_set<std::string>> allows; 439 std::unique_ptr<std::unordered_set<std::string>> allows;
440 std::unique_ptr<std::unordered_set<std::string>> globals;
441 }; 440 };
442 std::list<Scope> _scopes; 441 std::list<Scope> _scopes;
443 static const std::string Empty; 442 static const std::string Empty;
@@ -516,10 +515,8 @@ private:
516 int mode = int(std::isupper(name[0]) ? GlobalMode::Capital : GlobalMode::Any); 515 int mode = int(std::isupper(name[0]) ? GlobalMode::Capital : GlobalMode::Any);
517 const auto& current = _scopes.back(); 516 const auto& current = _scopes.back();
518 if (int(current.mode) >= mode) { 517 if (int(current.mode) >= mode) {
519 if (!current.globals) { 518 isDefined = true;
520 isDefined = true; 519 current.vars->insert_or_assign(name, VarType::Global);
521 current.vars->insert_or_assign(name, VarType::Global);
522 }
523 } 520 }
524 decltype(_scopes.back().allows.get()) allows = nullptr; 521 decltype(_scopes.back().allows.get()) allows = nullptr;
525 for (auto it = _scopes.rbegin(); it != _scopes.rend(); ++it) { 522 for (auto it = _scopes.rbegin(); it != _scopes.rend(); ++it) {
@@ -892,10 +889,6 @@ private:
892 void addGlobalVar(const std::string& name, ast_node* x) { 889 void addGlobalVar(const std::string& name, ast_node* x) {
893 if (isLocal(name)) throw CompileError("can not declare a local variable to be global"sv, x); 890 if (isLocal(name)) throw CompileError("can not declare a local variable to be global"sv, x);
894 auto& scope = _scopes.back(); 891 auto& scope = _scopes.back();
895 if (!scope.globals) {
896 scope.globals = std::make_unique<std::unordered_set<std::string>>();
897 }
898 scope.globals->insert(name);
899 scope.vars->insert_or_assign(name, VarType::Global); 892 scope.vars->insert_or_assign(name, VarType::Global);
900 } 893 }
901 894
@@ -1486,7 +1479,7 @@ private:
1486 } 1479 }
1487 } 1480 }
1488 1481
1489 bool isPureBackcall(Exp_t* exp) const { 1482 bool isPurePipeChain(Exp_t* exp) const {
1490 return exp->opValues.empty() && exp->pipeExprs.size() > 1; 1483 return exp->opValues.empty() && exp->pipeExprs.size() > 1;
1491 } 1484 }
1492 1485
@@ -1869,7 +1862,7 @@ private:
1869 } 1862 }
1870 } else if (expList->exprs.size() == 1) { 1863 } else if (expList->exprs.size() == 1) {
1871 auto exp = static_cast<Exp_t*>(expList->exprs.back()); 1864 auto exp = static_cast<Exp_t*>(expList->exprs.back());
1872 if (isPureBackcall(exp)) { 1865 if (isPurePipeChain(exp)) {
1873 transformExp(exp, out, ExpUsage::Common); 1866 transformExp(exp, out, ExpUsage::Common);
1874 break; 1867 break;
1875 } 1868 }
@@ -2061,7 +2054,7 @@ private:
2061 } 2054 }
2062 } 2055 }
2063 2056
2064 void transformAssignment(ExpListAssign_t* assignment, str_list& out, bool optionalDestruct = false) { 2057 bool transformAssignment(ExpListAssign_t* assignment, str_list& out, bool optionalDestruct = false) {
2065 checkAssignable(assignment->expList); 2058 checkAssignable(assignment->expList);
2066 BLOCK_START 2059 BLOCK_START
2067 auto assign = ast_cast<Assign_t>(assignment->action); 2060 auto assign = ast_cast<Assign_t>(assignment->action);
@@ -2176,7 +2169,7 @@ private:
2176 temp.push_back(indent() + "end"s + nll(assignment)); 2169 temp.push_back(indent() + "end"s + nll(assignment));
2177 } 2170 }
2178 out.push_back(join(temp)); 2171 out.push_back(join(temp));
2179 return; 2172 return false;
2180 BLOCK_END 2173 BLOCK_END
2181 } 2174 }
2182 { 2175 {
@@ -2248,7 +2241,7 @@ private:
2248 transformAssignment(afterAssignment, temp); 2241 transformAssignment(afterAssignment, temp);
2249 } 2242 }
2250 out.push_back(join(temp)); 2243 out.push_back(join(temp));
2251 return; 2244 return false;
2252 } else if (ast_is<TableAppendingOp_t>(chainValue->items.back())) { 2245 } else if (ast_is<TableAppendingOp_t>(chainValue->items.back())) {
2253 str_list temp; 2246 str_list temp;
2254 auto [beforeAssignment, afterAssignment] = splitAssignment(); 2247 auto [beforeAssignment, afterAssignment] = splitAssignment();
@@ -2300,7 +2293,7 @@ private:
2300 transformAssignment(afterAssignment, temp); 2293 transformAssignment(afterAssignment, temp);
2301 } 2294 }
2302 out.push_back(join(temp)); 2295 out.push_back(join(temp));
2303 return; 2296 return false;
2304 } else { 2297 } else {
2305 break; 2298 break;
2306 } 2299 }
@@ -2322,7 +2315,7 @@ private:
2322 std::string preDefine = getPreDefineLine(assignment); 2315 std::string preDefine = getPreDefineLine(assignment);
2323 transformIf(ifNode, out, ExpUsage::Assignment, assignList); 2316 transformIf(ifNode, out, ExpUsage::Assignment, assignList);
2324 out.back().insert(0, preDefine); 2317 out.back().insert(0, preDefine);
2325 return; 2318 return false;
2326 } 2319 }
2327 case id<Switch_t>(): { 2320 case id<Switch_t>(): {
2328 auto switchNode = static_cast<Switch_t*>(value); 2321 auto switchNode = static_cast<Switch_t*>(value);
@@ -2330,7 +2323,7 @@ private:
2330 std::string preDefine = getPreDefineLine(assignment); 2323 std::string preDefine = getPreDefineLine(assignment);
2331 transformSwitch(switchNode, out, ExpUsage::Assignment, assignList); 2324 transformSwitch(switchNode, out, ExpUsage::Assignment, assignList);
2332 out.back().insert(0, preDefine); 2325 out.back().insert(0, preDefine);
2333 return; 2326 return false;
2334 } 2327 }
2335 case id<With_t>(): { 2328 case id<With_t>(): {
2336 auto withNode = static_cast<With_t*>(value); 2329 auto withNode = static_cast<With_t*>(value);
@@ -2338,7 +2331,7 @@ private:
2338 std::string preDefine = getPreDefineLine(assignment); 2331 std::string preDefine = getPreDefineLine(assignment);
2339 transformWith(withNode, out, expList); 2332 transformWith(withNode, out, expList);
2340 out.back().insert(0, preDefine); 2333 out.back().insert(0, preDefine);
2341 return; 2334 return false;
2342 } 2335 }
2343 case id<Do_t>(): { 2336 case id<Do_t>(): {
2344 auto expList = assignment->expList.get(); 2337 auto expList = assignment->expList.get();
@@ -2346,7 +2339,7 @@ private:
2346 std::string preDefine = getPreDefineLine(assignment); 2339 std::string preDefine = getPreDefineLine(assignment);
2347 transformDo(doNode, out, ExpUsage::Assignment, expList); 2340 transformDo(doNode, out, ExpUsage::Assignment, expList);
2348 out.back().insert(0, preDefine); 2341 out.back().insert(0, preDefine);
2349 return; 2342 return false;
2350 } 2343 }
2351 case id<Comprehension_t>(): { 2344 case id<Comprehension_t>(): {
2352 auto comp = static_cast<Comprehension_t*>(value); 2345 auto comp = static_cast<Comprehension_t*>(value);
@@ -2358,42 +2351,42 @@ private:
2358 } else { 2351 } else {
2359 transformComprehension(comp, out, ExpUsage::Assignment, expList); 2352 transformComprehension(comp, out, ExpUsage::Assignment, expList);
2360 } 2353 }
2361 return; 2354 return false;
2362 } 2355 }
2363 case id<TblComprehension_t>(): { 2356 case id<TblComprehension_t>(): {
2364 auto expList = assignment->expList.get(); 2357 auto expList = assignment->expList.get();
2365 std::string preDefine = getPreDefineLine(assignment); 2358 std::string preDefine = getPreDefineLine(assignment);
2366 transformTblComprehension(static_cast<TblComprehension_t*>(value), out, ExpUsage::Assignment, expList); 2359 transformTblComprehension(static_cast<TblComprehension_t*>(value), out, ExpUsage::Assignment, expList);
2367 out.back().insert(0, preDefine); 2360 out.back().insert(0, preDefine);
2368 return; 2361 return false;
2369 } 2362 }
2370 case id<For_t>(): { 2363 case id<For_t>(): {
2371 auto expList = assignment->expList.get(); 2364 auto expList = assignment->expList.get();
2372 std::string preDefine = getPreDefineLine(assignment); 2365 std::string preDefine = getPreDefineLine(assignment);
2373 transformForInPlace(static_cast<For_t*>(value), out, expList); 2366 transformForInPlace(static_cast<For_t*>(value), out, expList);
2374 out.back().insert(0, preDefine); 2367 out.back().insert(0, preDefine);
2375 return; 2368 return false;
2376 } 2369 }
2377 case id<ForEach_t>(): { 2370 case id<ForEach_t>(): {
2378 auto expList = assignment->expList.get(); 2371 auto expList = assignment->expList.get();
2379 std::string preDefine = getPreDefineLine(assignment); 2372 std::string preDefine = getPreDefineLine(assignment);
2380 transformForEachInPlace(static_cast<ForEach_t*>(value), out, expList); 2373 transformForEachInPlace(static_cast<ForEach_t*>(value), out, expList);
2381 out.back().insert(0, preDefine); 2374 out.back().insert(0, preDefine);
2382 return; 2375 return false;
2383 } 2376 }
2384 case id<ClassDecl_t>(): { 2377 case id<ClassDecl_t>(): {
2385 auto expList = assignment->expList.get(); 2378 auto expList = assignment->expList.get();
2386 std::string preDefine = getPreDefineLine(assignment); 2379 std::string preDefine = getPreDefineLine(assignment);
2387 transformClassDecl(static_cast<ClassDecl_t*>(value), out, ExpUsage::Assignment, expList); 2380 transformClassDecl(static_cast<ClassDecl_t*>(value), out, ExpUsage::Assignment, expList);
2388 out.back().insert(0, preDefine); 2381 out.back().insert(0, preDefine);
2389 return; 2382 return false;
2390 } 2383 }
2391 case id<While_t>(): { 2384 case id<While_t>(): {
2392 auto expList = assignment->expList.get(); 2385 auto expList = assignment->expList.get();
2393 std::string preDefine = getPreDefineLine(assignment); 2386 std::string preDefine = getPreDefineLine(assignment);
2394 transformWhileInPlace(static_cast<While_t*>(value), out, expList); 2387 transformWhileInPlace(static_cast<While_t*>(value), out, expList);
2395 out.back().insert(0, preDefine); 2388 out.back().insert(0, preDefine);
2396 return; 2389 return false;
2397 } 2390 }
2398 case id<TableLit_t>(): { 2391 case id<TableLit_t>(): {
2399 auto tableLit = static_cast<TableLit_t*>(value); 2392 auto tableLit = static_cast<TableLit_t*>(value);
@@ -2402,7 +2395,7 @@ private:
2402 std::string preDefine = getPreDefineLine(assignment); 2395 std::string preDefine = getPreDefineLine(assignment);
2403 transformSpreadTable(tableLit->values.objects(), out, ExpUsage::Assignment, expList, false); 2396 transformSpreadTable(tableLit->values.objects(), out, ExpUsage::Assignment, expList, false);
2404 out.back().insert(0, preDefine); 2397 out.back().insert(0, preDefine);
2405 return; 2398 return false;
2406 } 2399 }
2407 break; 2400 break;
2408 } 2401 }
@@ -2413,31 +2406,31 @@ private:
2413 std::string preDefine = getPreDefineLine(assignment); 2406 std::string preDefine = getPreDefineLine(assignment);
2414 transformSpreadTable(tableBlock->values.objects(), out, ExpUsage::Assignment, expList, false); 2407 transformSpreadTable(tableBlock->values.objects(), out, ExpUsage::Assignment, expList, false);
2415 out.back().insert(0, preDefine); 2408 out.back().insert(0, preDefine);
2416 return; 2409 return false;
2417 } 2410 }
2418 break; 2411 break;
2419 } 2412 }
2420 } 2413 }
2421 auto exp = ast_cast<Exp_t>(value); 2414 auto exp = ast_cast<Exp_t>(value);
2422 BREAK_IF(!exp); 2415 BREAK_IF(!exp);
2423 if (isPureBackcall(exp)) { 2416 if (isPurePipeChain(exp)) {
2424 auto expList = assignment->expList.get(); 2417 auto expList = assignment->expList.get();
2425 transformExp(exp, out, ExpUsage::Assignment, expList); 2418 transformExp(exp, out, ExpUsage::Assignment, expList);
2426 return; 2419 return false;
2427 } else if (isPureNilCoalesed(exp)) { 2420 } else if (isPureNilCoalesed(exp)) {
2428 auto expList = assignment->expList.get(); 2421 auto expList = assignment->expList.get();
2429 transformNilCoalesedExp(exp, out, ExpUsage::Assignment, expList); 2422 transformNilCoalesedExp(exp, out, ExpUsage::Assignment, expList);
2430 return; 2423 return false;
2431 } else if (auto unary = unaryGeneratingAnonFunc(exp)) { 2424 } else if (auto unary = unaryGeneratingAnonFunc(exp)) {
2432 std::string preDefine = getPreDefineLine(assignment); 2425 std::string preDefine = getPreDefineLine(assignment);
2433 auto expList = assignment->expList.get(); 2426 auto expList = assignment->expList.get();
2434 transformUnaryExp(unary, out, ExpUsage::Assignment, expList); 2427 transformUnaryExp(unary, out, ExpUsage::Assignment, expList);
2435 out.back().insert(0, preDefine); 2428 out.back().insert(0, preDefine);
2436 return; 2429 return false;
2437 } else if (isConditionChaining(exp)) { 2430 } else if (isConditionChaining(exp)) {
2438 auto expList = assignment->expList.get(); 2431 auto expList = assignment->expList.get();
2439 transformExp(exp, out, ExpUsage::Assignment, expList); 2432 transformExp(exp, out, ExpUsage::Assignment, expList);
2440 return; 2433 return false;
2441 } 2434 }
2442 auto singleVal = singleValueFrom(exp); 2435 auto singleVal = singleValueFrom(exp);
2443 BREAK_IF(!singleVal); 2436 BREAK_IF(!singleVal);
@@ -2451,13 +2444,13 @@ private:
2451 std::string preDefine = getPreDefineLine(assignment); 2444 std::string preDefine = getPreDefineLine(assignment);
2452 transformChainValue(chainValue, out, ExpUsage::Assignment, expList, false, optionalDestruct); 2445 transformChainValue(chainValue, out, ExpUsage::Assignment, expList, false, optionalDestruct);
2453 out.back().insert(0, preDefine); 2446 out.back().insert(0, preDefine);
2454 return; 2447 return false;
2455 } 2448 }
2456 case ChainType::HasKeyword: 2449 case ChainType::HasKeyword:
2457 case ChainType::HasUnicode: 2450 case ChainType::HasUnicode:
2458 case ChainType::Macro: 2451 case ChainType::Macro:
2459 transformChainValue(chainValue, out, ExpUsage::Assignment, expList); 2452 transformChainValue(chainValue, out, ExpUsage::Assignment, expList);
2460 return; 2453 return false;
2461 case ChainType::Common: 2454 case ChainType::Common:
2462 case ChainType::EndWithEOP: 2455 case ChainType::EndWithEOP:
2463 case ChainType::Metatable: 2456 case ChainType::Metatable:
@@ -2468,6 +2461,7 @@ private:
2468 auto info = extractDestructureInfo(assignment, false, optionalDestruct); 2461 auto info = extractDestructureInfo(assignment, false, optionalDestruct);
2469 if (info.destructures.empty()) { 2462 if (info.destructures.empty()) {
2470 transformAssignmentCommon(assignment, out); 2463 transformAssignmentCommon(assignment, out);
2464 return true;
2471 } else { 2465 } else {
2472 auto x = assignment; 2466 auto x = assignment;
2473 str_list temp; 2467 str_list temp;
@@ -2733,6 +2727,7 @@ private:
2733 } 2727 }
2734 out.push_back(join(temp)); 2728 out.push_back(join(temp));
2735 } 2729 }
2730 return false;
2736 } 2731 }
2737 2732
2738 void transformAssignItem(ast_node* value, str_list& out) { 2733 void transformAssignItem(ast_node* value, str_list& out) {
@@ -4800,8 +4795,8 @@ private:
4800 auto varName = variableToString(ast_to<Variable_t>(var)); 4795 auto varName = variableToString(ast_to<Variable_t>(var));
4801 auto closeVar = getUnusedName("_close_"sv); 4796 auto closeVar = getUnusedName("_close_"sv);
4802 addToScope(closeVar); 4797 addToScope(closeVar);
4803 getCloses.push_back(closeVar + "=assert "s + varName + ".<close>"s); 4798 getCloses.push_back(closeVar + "=if type("s + varName + ") in ['table', 'userdata'] then assert "s + varName + ".<> and "s + varName +".<close>, \""s + "variable '"s + varName + "' got a non-closable value\" elseif "s + varName + " == nil then nil else error \""s + "variable '"s + varName + "' got a non-closable value\"");
4804 doCloses.push_front(closeVar + ' ' + varName); 4799 doCloses.push_front(closeVar + "? "s + varName);
4805 } 4800 }
4806 popScope(); 4801 popScope();
4807 auto okVar = getUnusedName("_ok_"sv); 4802 auto okVar = getUnusedName("_ok_"sv);
@@ -5313,7 +5308,7 @@ private:
5313 if (auto valueList = returnNode->valueList.as<ExpListLow_t>()) { 5308 if (auto valueList = returnNode->valueList.as<ExpListLow_t>()) {
5314 if (valueList->exprs.size() == 1) { 5309 if (valueList->exprs.size() == 1) {
5315 auto exp = static_cast<Exp_t*>(valueList->exprs.back()); 5310 auto exp = static_cast<Exp_t*>(valueList->exprs.back());
5316 if (isPureBackcall(exp)) { 5311 if (isPurePipeChain(exp)) {
5317 transformExp(exp, out, ExpUsage::Return); 5312 transformExp(exp, out, ExpUsage::Return);
5318 return; 5313 return;
5319 } else if (isPureNilCoalesed(exp)) { 5314 } else if (isPureNilCoalesed(exp)) {
@@ -9399,7 +9394,6 @@ private:
9399 auto classDecl = static_cast<ClassDecl_t*>(item); 9394 auto classDecl = static_cast<ClassDecl_t*>(item);
9400 if (classDecl->name) { 9395 if (classDecl->name) {
9401 if (auto var = classDecl->name->item.as<Variable_t>()) { 9396 if (auto var = classDecl->name->item.as<Variable_t>()) {
9402 markVarsGlobal(GlobalMode::Any);
9403 addGlobalVar(variableToString(var), classDecl->name->item); 9397 addGlobalVar(variableToString(var), classDecl->name->item);
9404 } 9398 }
9405 } 9399 }
@@ -9414,7 +9408,6 @@ private:
9414 } 9408 }
9415 break; 9409 break;
9416 case id<GlobalValues_t>(): { 9410 case id<GlobalValues_t>(): {
9417 markVarsGlobal(GlobalMode::Any);
9418 auto values = global->item.to<GlobalValues_t>(); 9411 auto values = global->item.to<GlobalValues_t>();
9419 if (values->valueList) { 9412 if (values->valueList) {
9420 auto expList = x->new_ptr<ExpList_t>(); 9413 auto expList = x->new_ptr<ExpList_t>();
@@ -10676,6 +10669,7 @@ private:
10676 10669
10677 void transformLocal(Local_t* local, str_list& out) { 10670 void transformLocal(Local_t* local, str_list& out) {
10678 str_list temp; 10671 str_list temp;
10672 bool defined = local->defined;
10679 if (!local->defined) { 10673 if (!local->defined) {
10680 local->defined = true; 10674 local->defined = true;
10681 transformLocalDef(local, temp); 10675 transformLocalDef(local, temp);
@@ -10704,7 +10698,26 @@ private:
10704 assign->values.push_back(tableBlock); 10698 assign->values.push_back(tableBlock);
10705 } 10699 }
10706 assignment->action.set(assign); 10700 assignment->action.set(assign);
10707 transformAssignment(assignment, temp); 10701 bool oneLined = transformAssignment(assignment, temp);
10702 for (auto val : assign->values.objects()) {
10703 if (auto value = singleValueFrom(val)) {
10704 if (auto spValue = value->item.as<SimpleValue_t>()) {
10705 if (auto funLit = spValue->value.as<FunLit_t>()) {
10706 if (!funLit->noRecursion) {
10707 oneLined = false;
10708 }
10709 break;
10710 }
10711 }
10712 }
10713 }
10714 if (!defined && oneLined && temp.size() == 2) {
10715 auto pos = temp.back().find_first_not_of(" \t"sv);
10716 if (pos != std::string::npos) {
10717 temp.back().insert(pos, "local ");
10718 temp.pop_front();
10719 }
10720 }
10708 } 10721 }
10709 break; 10722 break;
10710 } 10723 }
diff --git a/src/yuescript/yue_parser.cpp b/src/yuescript/yue_parser.cpp
index 5993ed5..ceb1f7c 100644
--- a/src/yuescript/yue_parser.cpp
+++ b/src/yuescript/yue_parser.cpp
@@ -519,7 +519,7 @@ YueParser::YueParser() {
519 UpdateOp = 519 UpdateOp =
520 expr("..") | "//" | "or" | "and" | 520 expr("..") | "//" | "or" | "and" |
521 ">>" | "<<" | "??" | 521 ">>" | "<<" | "??" |
522 set("+-*/%&|"); 522 set("+-*/%&|^");
523 523
524 Update = UpdateOp >> '=' >> space >> Exp; 524 Update = UpdateOp >> '=' >> space >> Exp;
525 525
diff --git a/src/yuescript/yuescript.h b/src/yuescript/yuescript.h
index ebe90cb..83a4350 100644
--- a/src/yuescript/yuescript.h
+++ b/src/yuescript/yuescript.h
@@ -187,10 +187,10 @@ local function dump(what)
187 depth = depth or 0 187 depth = depth or 0
188 local t = type(what) 188 local t = type(what)
189 if "string" == t then 189 if "string" == t then
190 return "\"" .. tostring(what) .. "\"\n" 190 return "\"" .. tostring(what) .. "\""
191 elseif "table" == t then 191 elseif "table" == t then
192 if seen[what] then 192 if seen[what] then
193 return "recursion(" .. tostring(what) .. ")...\n" 193 return "recursion(" .. tostring(what) .. ")..."
194 end 194 end
195 seen[what] = true 195 seen[what] = true
196 depth = depth + 1 196 depth = depth + 1
@@ -199,9 +199,9 @@ local function dump(what)
199 insert(lines, ('\t'):rep(depth) .. "[" .. tostring(k) .. "] = " .. _dump(v, depth)) 199 insert(lines, ('\t'):rep(depth) .. "[" .. tostring(k) .. "] = " .. _dump(v, depth))
200 end 200 end
201 seen[what] = false 201 seen[what] = false
202 return "{\n" .. concat(lines) .. ('\t'):rep(depth - 1) .. "}\n" 202 return "{\n" .. concat(lines, "\n") .. "\n" .. ('\t'):rep(depth - 1) .. "}"
203 else 203 else
204 return tostring(what) .. "\n" 204 return tostring(what)
205 end 205 end
206 end 206 end
207 return _dump(what) 207 return _dump(what)