/* * Copyright (c) 2022, Brandon Pruitt * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include ErrorOr serenity_main(Main::Arguments arguments) { TRY(Core::System::pledge("stdio rpath inet")); TRY(Core::System::unveil("/sys/kernel/net", "r")); TRY(Core::System::unveil(nullptr, nullptr)); StringView modify_action; StringView value_host_address; StringView value_network_address; StringView value_gateway_address; StringView value_netmask_address; StringView value_interface; Core::ArgsParser args_parser; args_parser.set_general_help("Display kernel routing table"); args_parser.add_positional_argument(modify_action, "Modify the global routing table { add | del }", "action", Core::ArgsParser::Required::No); args_parser.add_option(value_host_address, "Target destination is an IPv4 address", "host", 'h', "host"); args_parser.add_option(value_network_address, "Target destination is a network address", "net", 'n', "net"); args_parser.add_option(value_gateway_address, "Route packets via a gateway", "gw", 'g', "gw"); args_parser.add_option(value_netmask_address, "The netmask to be used when adding a network route", "netmask", 'm', "netmask"); args_parser.add_option(value_interface, "Force the route to be associated with the specified device interface", "interface", 'i', "interface"); args_parser.parse(arguments); enum class Alignment { Left, Right }; struct Column { DeprecatedString title; Alignment alignment { Alignment::Left }; int width { 0 }; DeprecatedString buffer; }; Vector columns; int destination_column = -1; int gateway_column = -1; int genmask_column = -1; int flags_column = -1; int interface_column = -1; auto add_column = [&](auto title, auto alignment, auto width) { columns.append({ title, alignment, width, {} }); return columns.size() - 1; }; destination_column = add_column("Destination", Alignment::Left, 15); gateway_column = add_column("Gateway", Alignment::Left, 15); genmask_column = add_column("Genmask", Alignment::Left, 15); flags_column = add_column("Flags", Alignment::Left, 5); interface_column = add_column("Interface", Alignment::Left, 9); auto print_column = [](auto& column, auto& string) { if (!column.width) { out("{}", string); return; } if (column.alignment == Alignment::Right) { out("{:>{1}} "sv, string, column.width); } else { out("{:<{1}} "sv, string, column.width); } }; if (modify_action.is_empty()) { auto file = TRY(Core::File::open("/sys/kernel/net/route"sv, Core::File::OpenMode::Read)); auto file_contents = TRY(file->read_until_eof()); auto json = TRY(JsonValue::from_string(file_contents)); outln("Kernel IP routing table"); for (auto& column : columns) print_column(column, column.title); outln(); Vector sorted_regions = json.as_array().values(); quick_sort(sorted_regions, [](auto& a, auto& b) { return a.as_object().get_deprecated_string("destination"sv).value_or({}) < b.as_object().get_deprecated_string("destination"sv).value_or({}); }); for (auto& value : sorted_regions) { auto& if_object = value.as_object(); auto destination = if_object.get_deprecated_string("destination"sv).value_or({}); auto gateway = if_object.get_deprecated_string("gateway"sv).value_or({}); auto genmask = if_object.get_deprecated_string("genmask"sv).value_or({}); auto interface = if_object.get_deprecated_string("interface"sv).value_or({}); auto flags = if_object.get_u32("flags"sv).value_or(0); StringBuilder flags_builder; if (flags & RTF_UP) flags_builder.append('U'); if (flags & RTF_GATEWAY) flags_builder.append('G'); if (flags & RTF_HOST) flags_builder.append('H'); if (destination_column != -1) columns[destination_column].buffer = destination; if (gateway_column != -1) columns[gateway_column].buffer = gateway; if (genmask_column != -1) columns[genmask_column].buffer = genmask; if (flags_column != -1) columns[flags_column].buffer = flags_builder.string_view(); if (interface_column != -1) columns[interface_column].buffer = interface; for (auto& column : columns) print_column(column, column.buffer); outln(); }; } if (!modify_action.is_empty()) { bool const action_add = (modify_action == "add"); bool const action_del = (modify_action == "del"); if (!action_add && !action_del) { warnln("Invalid modify action: {}", modify_action); return 1; } if (value_host_address.is_empty() && value_network_address.is_empty()) { warnln("No target host or network specified"); return 1; } Optional destination; if (!value_host_address.is_empty()) destination = AK::IPv4Address::from_string(value_host_address); StringView address; StringView cidr; if (!value_network_address.is_empty()) { // Check if a CIDR notation was provided and parse accordingly if (auto position = value_network_address.find('/'); position.has_value()) { address = value_network_address.substring_view(0, position.value()); cidr = value_network_address.substring_view(position.value() + 1); } else { address = value_network_address; } destination = AK::IPv4Address::from_string(address); } if (!destination.has_value()) { warnln("Invalid destination IPv4 address"); return 1; } auto gateway = AK::IPv4Address::from_string(value_gateway_address); if (action_add && !gateway.has_value()) { warnln("Invalid gateway IPv4 address: '{}'", value_gateway_address); return 1; } Optional genmask; if (auto cidr_int = cidr.to_int(); cidr_int.has_value()) genmask = AK::IPv4Address::netmask_from_cidr(cidr_int.value()); else genmask = AK::IPv4Address::from_string(value_netmask_address); if (!genmask.has_value()) { warnln("Invalid genmask IPv4 address: '{}'", value_netmask_address); return 1; } int fd = TRY(Core::System::socket(AF_INET, SOCK_DGRAM, IPPROTO_IP)); rtentry rt {}; memset(&rt, 0, sizeof(rt)); rt.rt_dev = const_cast(value_interface.characters_without_null_termination()); rt.rt_gateway.sa_family = AF_INET; ((sockaddr_in&)rt.rt_dst).sin_addr.s_addr = destination.value().to_in_addr_t(); ((sockaddr_in&)rt.rt_gateway).sin_addr.s_addr = gateway.value_or(IPv4Address {}).to_in_addr_t(); ((sockaddr_in&)rt.rt_genmask).sin_addr.s_addr = genmask.value().to_in_addr_t(); rt.rt_flags = RTF_UP; if (!value_host_address.is_empty()) rt.rt_flags |= RTF_HOST; rt.rt_flags |= RTF_GATEWAY; if (action_add) TRY(Core::System::ioctl(fd, SIOCADDRT, &rt)); if (action_del) TRY(Core::System::ioctl(fd, SIOCDELRT, &rt)); } return 0; }