GenerateCSSMathFunctions.cpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379
  1. /*
  2. * Copyright (c) 2022-2023, Sam Atkins <atkinssj@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include "GeneratorUtil.h"
  7. #include <AK/SourceGenerator.h>
  8. #include <AK/StringBuilder.h>
  9. #include <LibCore/ArgsParser.h>
  10. #include <LibMain/Main.h>
  11. namespace {
  12. ErrorOr<void> generate_header_file(JsonObject& functions_data, Core::File& file)
  13. {
  14. StringBuilder builder;
  15. SourceGenerator generator { builder };
  16. generator.append(R"~~~(
  17. // This file is generated by GenerateCSSMathFunctions.cpp
  18. #pragma once
  19. namespace Web::CSS {
  20. enum class MathFunction {
  21. )~~~");
  22. functions_data.for_each_member([&](auto& name, auto&) {
  23. auto member_generator = generator.fork();
  24. member_generator.set("name:titlecase", title_casify(name));
  25. member_generator.appendln(" @name:titlecase@,"sv);
  26. });
  27. generator.append(R"~~~(
  28. };
  29. }
  30. )~~~");
  31. TRY(file.write_until_depleted(generator.as_string_view().bytes()));
  32. return {};
  33. }
  34. String generate_calculation_type_check(StringView calculation_variable_name, StringView parameter_types)
  35. {
  36. StringBuilder builder;
  37. auto allowed_types = parameter_types.split_view('|');
  38. bool first_type_check = true;
  39. for (auto const& allowed_type_name : allowed_types) {
  40. if (!first_type_check)
  41. builder.append(" || "sv);
  42. first_type_check = false;
  43. if (allowed_type_name == "<angle>"sv) {
  44. builder.appendff("{}.{}", calculation_variable_name, "matches_angle()"sv);
  45. } else if (allowed_type_name == "<dimension>"sv) {
  46. builder.appendff("{}.{}", calculation_variable_name, "matches_dimension()"sv);
  47. } else if (allowed_type_name == "<flex>"sv) {
  48. builder.appendff("{}.{}", calculation_variable_name, "matches_flex()"sv);
  49. } else if (allowed_type_name == "<frequency>"sv) {
  50. builder.appendff("{}.{}", calculation_variable_name, "matches_frequency()"sv);
  51. } else if (allowed_type_name == "<length>"sv) {
  52. builder.appendff("{}.{}", calculation_variable_name, "matches_length()"sv);
  53. } else if (allowed_type_name == "<number>"sv) {
  54. builder.appendff("{}.{}", calculation_variable_name, "matches_number()"sv);
  55. } else if (allowed_type_name == "<percentage>"sv) {
  56. builder.appendff("{}.{}", calculation_variable_name, "matches_percentage()"sv);
  57. } else if (allowed_type_name == "<resolution>"sv) {
  58. builder.appendff("{}.{}", calculation_variable_name, "matches_resolution()"sv);
  59. } else if (allowed_type_name == "<time>"sv) {
  60. builder.appendff("{}.{}", calculation_variable_name, "matches_time()"sv);
  61. } else {
  62. dbgln("I don't know what '{}' is!", allowed_type_name);
  63. VERIFY_NOT_REACHED();
  64. }
  65. }
  66. return MUST(builder.to_string());
  67. }
  68. ErrorOr<void> generate_implementation_file(JsonObject& functions_data, Core::File& file)
  69. {
  70. StringBuilder builder;
  71. SourceGenerator generator { builder };
  72. generator.append(R"~~~(
  73. // This file is generated by GenerateCSSMathFunctions.cpp
  74. #include <AK/Debug.h>
  75. #include <LibWeb/CSS/MathFunctions.h>
  76. #include <LibWeb/CSS/Parser/Parser.h>
  77. #include <LibWeb/CSS/Enums.h>
  78. #include <LibWeb/CSS/StyleValues/CalculatedStyleValue.h>
  79. namespace Web::CSS::Parser {
  80. static Optional<RoundingStrategy> parse_rounding_strategy(Vector<ComponentValue> const& tokens)
  81. {
  82. auto stream = TokenStream { tokens };
  83. stream.skip_whitespace();
  84. if (!stream.has_next_token())
  85. return {};
  86. auto& ident = stream.next_token();
  87. if (!ident.is(Token::Type::Ident))
  88. return {};
  89. stream.skip_whitespace();
  90. if (stream.has_next_token())
  91. return {};
  92. auto maybe_identifier = value_id_from_string(ident.token().ident());
  93. if (!maybe_identifier.has_value())
  94. return {};
  95. return value_id_to_rounding_strategy(maybe_identifier.value());
  96. }
  97. OwnPtr<CalculationNode> Parser::parse_math_function(PropertyID property_id, Function const& function)
  98. {
  99. TokenStream stream { function.values() };
  100. auto arguments = parse_a_comma_separated_list_of_component_values(stream);
  101. )~~~");
  102. functions_data.for_each_member([&](auto& name, JsonValue const& value) -> void {
  103. auto& function_data = value.as_object();
  104. auto& parameters = function_data.get_array("parameters"sv).value();
  105. auto function_generator = generator.fork();
  106. function_generator.set("name:lowercase", name);
  107. function_generator.set("name:titlecase", title_casify(name));
  108. function_generator.appendln(" if (function.name().equals_ignoring_ascii_case(\"@name:lowercase@\"sv)) {");
  109. if (function_data.get_bool("is-variadic"sv).value_or(false)) {
  110. // Variadic function
  111. function_generator.append(R"~~~(
  112. CSSNumericType determined_argument_type;
  113. Vector<NonnullOwnPtr<CalculationNode>> parsed_arguments;
  114. parsed_arguments.ensure_capacity(arguments.size());
  115. for (auto& argument : arguments) {
  116. auto calculation_node = parse_a_calculation(argument);
  117. if (!calculation_node) {
  118. dbgln_if(CSS_PARSER_DEBUG, "@name:lowercase@() argument #{} is not a valid calculation", parsed_arguments.size());
  119. return nullptr;
  120. }
  121. auto maybe_argument_type = calculation_node->determine_type(m_context.current_property_id());
  122. if (!maybe_argument_type.has_value()) {
  123. dbgln_if(CSS_PARSER_DEBUG, "@name:lowercase@() argument #{} couldn't determine its type", parsed_arguments.size());
  124. return nullptr;
  125. }
  126. auto argument_type = maybe_argument_type.release_value();
  127. )~~~");
  128. // Generate some type checks
  129. VERIFY(parameters.size() == 1);
  130. auto& parameter_data = parameters[0].as_object();
  131. auto parameter_type_string = parameter_data.get_byte_string("type"sv).value();
  132. function_generator.set("type_check", generate_calculation_type_check("argument_type"sv, parameter_type_string));
  133. function_generator.append(R"~~~(
  134. if (!(@type_check@)) {
  135. dbgln_if(CSS_PARSER_DEBUG, "@name:lowercase@() argument #{} type ({}) is not an accepted type", parsed_arguments.size(), MUST(argument_type.dump()));
  136. return nullptr;
  137. }
  138. if (parsed_arguments.is_empty()) {
  139. determined_argument_type = move(argument_type);
  140. } else {
  141. if (determined_argument_type != argument_type) {
  142. dbgln_if(CSS_PARSER_DEBUG, "@name:lowercase@() argument #{} type ({}) doesn't match type of previous arguments ({})", parsed_arguments.size(), MUST(argument_type.dump()), MUST(determined_argument_type.dump()));
  143. return nullptr;
  144. }
  145. }
  146. parsed_arguments.append(calculation_node.release_nonnull());
  147. }
  148. return @name:titlecase@CalculationNode::create(move(parsed_arguments));
  149. }
  150. )~~~");
  151. } else {
  152. // Function with specified parameters.
  153. size_t min_argument_count = 0;
  154. size_t max_argument_count = parameters.size();
  155. parameters.for_each([&](JsonValue const& parameter_value) {
  156. auto& parameter = parameter_value.as_object();
  157. if (parameter.get_bool("required"sv) == true)
  158. min_argument_count++;
  159. });
  160. function_generator.set("min_argument_count", MUST(String::number(min_argument_count)));
  161. function_generator.set("max_argument_count", MUST(String::number(max_argument_count)));
  162. function_generator.append(R"~~~(
  163. if (arguments.size() < @min_argument_count@ || arguments.size() > @max_argument_count@) {
  164. dbgln_if(CSS_PARSER_DEBUG, "@name:lowercase@() has wrong number of arguments {}, expected between @min_argument_count@ and @max_argument_count@ inclusive", arguments.size());
  165. return nullptr;
  166. }
  167. size_t argument_index = 0;
  168. [[maybe_unused]] CSSNumericType previous_argument_type;
  169. )~~~");
  170. size_t parameter_index = 0;
  171. StringView previous_parameter_type_string;
  172. parameters.for_each([&](JsonValue const& parameter_value) {
  173. auto& parameter = parameter_value.as_object();
  174. auto parameter_type_string = parameter.get_byte_string("type"sv).value();
  175. auto parameter_required = parameter.get_bool("required"sv).value();
  176. auto parameter_generator = function_generator.fork();
  177. parameter_generator.set("parameter_name", parameter.get_byte_string("name"sv).value());
  178. parameter_generator.set("parameter_index", MUST(String::number(parameter_index)));
  179. bool parameter_is_calculation;
  180. if (parameter_type_string == "<rounding-strategy>") {
  181. parameter_is_calculation = false;
  182. parameter_generator.set("parameter_type", "RoundingStrategy"_string);
  183. parameter_generator.set("parse_function", "parse_rounding_strategy(arguments[argument_index])"_string);
  184. parameter_generator.set("check_function", ".has_value()"_string);
  185. parameter_generator.set("release_function", ".release_value()"_string);
  186. if (auto default_value = parameter.get_byte_string("default"sv); default_value.has_value()) {
  187. parameter_generator.set("parameter_default", MUST(String::formatted(" = RoundingStrategy::{}", title_casify(default_value.value()))));
  188. } else {
  189. parameter_generator.set("parameter_default", ""_string);
  190. }
  191. } else {
  192. // NOTE: This assumes everything not handled above is a calculation node of some kind.
  193. parameter_is_calculation = true;
  194. parameter_generator.set("parameter_type", "OwnPtr<CalculationNode>"_string);
  195. parameter_generator.set("parse_function", "parse_a_calculation(arguments[argument_index])"_string);
  196. parameter_generator.set("check_function", " != nullptr"_string);
  197. parameter_generator.set("release_function", ".release_nonnull()"_string);
  198. // NOTE: We have exactly one default value in the data right now, and it's a `<calc-constant>`,
  199. // so that's all we handle.
  200. if (auto default_value = parameter.get_byte_string("default"sv); default_value.has_value()) {
  201. parameter_generator.set("parameter_default", MUST(String::formatted(" = ConstantCalculationNode::create(CalculationNode::constant_type_from_string(\"{}\"sv).value())", default_value.value())));
  202. } else {
  203. parameter_generator.set("parameter_default", ""_string);
  204. }
  205. }
  206. parameter_generator.append(R"~~~(
  207. @parameter_type@ parameter_@parameter_index@@parameter_default@;
  208. )~~~");
  209. if (parameter_required) {
  210. parameter_generator.append(R"~~~(
  211. if (argument_index >= arguments.size()) {
  212. dbgln_if(CSS_PARSER_DEBUG, "@name:lowercase@() missing required argument '@parameter_name@'");
  213. return nullptr;
  214. } else {
  215. )~~~");
  216. } else {
  217. parameter_generator.append(R"~~~(
  218. if (argument_index < arguments.size()) {
  219. )~~~");
  220. }
  221. parameter_generator.append(R"~~~(
  222. auto maybe_parsed_argument_@parameter_index@ = @parse_function@;
  223. if (maybe_parsed_argument_@parameter_index@@check_function@) {
  224. parameter_@parameter_index@ = maybe_parsed_argument_@parameter_index@@release_function@;
  225. argument_index++;
  226. )~~~");
  227. if (parameter_required) {
  228. parameter_generator.append(R"~~~(
  229. } else {
  230. dbgln_if(CSS_PARSER_DEBUG, "@name:lowercase@() required argument '@parameter_name@' failed to parse");
  231. return nullptr;
  232. )~~~");
  233. }
  234. parameter_generator.append(R"~~~(
  235. }
  236. }
  237. )~~~");
  238. if (parameter_is_calculation) {
  239. auto parameter_type_variable = MUST(String::formatted("argument_type_{}", parameter_index));
  240. parameter_generator.set("type_check", generate_calculation_type_check(parameter_type_variable, parameter_type_string));
  241. parameter_generator.append(R"~~~(
  242. auto maybe_argument_type_@parameter_index@ = parameter_@parameter_index@->determine_type(property_id);
  243. if (!maybe_argument_type_@parameter_index@.has_value()) {
  244. dbgln_if(CSS_PARSER_DEBUG, "@name:lowercase@() argument '@parameter_name@' couldn't determine its type");
  245. return nullptr;
  246. }
  247. auto argument_type_@parameter_index@ = maybe_argument_type_@parameter_index@.release_value();
  248. if (!(@type_check@)) {
  249. dbgln_if(CSS_PARSER_DEBUG, "@name:lowercase@() argument '@parameter_name@' type ({}) is not an accepted type", MUST(argument_type_@parameter_index@.dump()));
  250. return nullptr;
  251. }
  252. )~~~");
  253. // NOTE: In all current cases, the parameters that take the same types must resolve to the same CSSNumericType.
  254. // This is a bit of a hack, but serves our needs for now.
  255. if (previous_parameter_type_string == parameter_type_string) {
  256. parameter_generator.append(R"~~~(
  257. if (argument_type_@parameter_index@ != previous_argument_type) {
  258. dbgln_if(CSS_PARSER_DEBUG, "@name:lowercase@() argument '@parameter_name@' type ({}) doesn't match type of previous arguments ({})", MUST(argument_type_@parameter_index@.dump()), MUST(previous_argument_type.dump()));
  259. return nullptr;
  260. }
  261. )~~~");
  262. }
  263. parameter_generator.append(R"~~~(
  264. previous_argument_type = argument_type_@parameter_index@;
  265. )~~~");
  266. }
  267. parameter_index++;
  268. previous_parameter_type_string = parameter_type_string;
  269. });
  270. // Generate the call to the constructor
  271. function_generator.append(" return @name:titlecase@CalculationNode::create("sv);
  272. parameter_index = 0;
  273. parameters.for_each([&](JsonValue const& parameter_value) {
  274. auto& parameter = parameter_value.as_object();
  275. auto parameter_type_string = parameter.get_byte_string("type"sv).value();
  276. auto parameter_generator = function_generator.fork();
  277. parameter_generator.set("parameter_index"sv, MUST(String::number(parameter_index)));
  278. if (parameter_type_string == "<rounding-strategy>"sv) {
  279. parameter_generator.set("release_value"sv, ""_string);
  280. } else {
  281. // NOTE: This assumes everything not handled above is a calculation node of some kind.
  282. parameter_generator.set("release_value"sv, ".release_nonnull()"_string);
  283. }
  284. if (parameter_index == 0) {
  285. parameter_generator.append("parameter_@parameter_index@@release_value@"sv);
  286. } else {
  287. parameter_generator.append(", parameter_@parameter_index@@release_value@"sv);
  288. }
  289. parameter_index++;
  290. });
  291. function_generator.append(R"~~~();
  292. }
  293. )~~~");
  294. }
  295. });
  296. generator.append(R"~~~(
  297. return nullptr;
  298. }
  299. }
  300. )~~~");
  301. TRY(file.write_until_depleted(generator.as_string_view().bytes()));
  302. return {};
  303. }
  304. } // end anonymous namespace
  305. ErrorOr<int> serenity_main(Main::Arguments arguments)
  306. {
  307. StringView generated_header_path;
  308. StringView generated_implementation_path;
  309. StringView identifiers_json_path;
  310. Core::ArgsParser args_parser;
  311. args_parser.add_option(generated_header_path, "Path to the MathFunctions header file to generate", "generated-header-path", 'h', "generated-header-path");
  312. args_parser.add_option(generated_implementation_path, "Path to the MathFunctions implementation file to generate", "generated-implementation-path", 'c', "generated-implementation-path");
  313. args_parser.add_option(identifiers_json_path, "Path to the JSON file to read from", "json-path", 'j', "json-path");
  314. args_parser.parse(arguments);
  315. auto json = TRY(read_entire_file_as_json(identifiers_json_path));
  316. VERIFY(json.is_object());
  317. auto math_functions_data = json.as_object();
  318. auto generated_header_file = TRY(Core::File::open(generated_header_path, Core::File::OpenMode::Write));
  319. auto generated_implementation_file = TRY(Core::File::open(generated_implementation_path, Core::File::OpenMode::Write));
  320. TRY(generate_header_file(math_functions_data, *generated_header_file));
  321. TRY(generate_implementation_file(math_functions_data, *generated_implementation_file));
  322. return 0;
  323. }