/* * Copyright (c) 2008 Patrick McHardy * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. * * Development of this code funded by Astaro AG (http://www.astaro.com/) */ #include #include #include #include #include #include #include #include #include #include #include #include void datatype_print(const struct expr *expr) { const struct datatype *dtype = expr->dtype; if (dtype->print != NULL) return dtype->print(expr); if (dtype->sym_tbl != NULL) return symbolic_constant_print(dtype->sym_tbl, expr); BUG(); } struct error_record *symbol_parse(const struct expr *sym, struct expr **res) { const struct datatype *dtype = sym->sym_type; assert(sym->ops->type == EXPR_SYMBOL); if (dtype == NULL) return error(&sym->location, "No symbol type information"); if (dtype->parse != NULL) return dtype->parse(sym, res); if (dtype->sym_tbl != NULL) return symbolic_constant_parse(sym, dtype->sym_tbl, res); return error(&sym->location, "Can't parse symbolic %s expressions", sym->sym_type->name); } struct error_record *symbolic_constant_parse(const struct expr *sym, const struct symbol_table *tbl, struct expr **res) { const struct symbolic_constant *s; for (s = tbl->symbols; s->identifier != NULL; s++) { if (!strcmp(sym->identifier, s->identifier)) break; } if (s->identifier == NULL) return error(&sym->location, "Could not parse %s", sym->sym_type->name); *res = constant_expr_alloc(&sym->location, sym->sym_type, tbl->byteorder, tbl->size, &s->value); return NULL; } void symbolic_constant_print(const struct symbol_table *tbl, const struct expr *expr) { const struct symbolic_constant *s; for (s = tbl->symbols; s->identifier != NULL; s++) { if (!mpz_cmp_ui(expr->value, s->value)) break; } if (s->identifier == NULL) return expr_basetype(expr)->print(expr); printf("%s", s->identifier); } void symbol_table_print(const struct symbol_table *tbl) { const struct symbolic_constant *s; unsigned int size = 2 * tbl->size / BITS_PER_BYTE; for (s = tbl->symbols; s->identifier != NULL; s++) printf("\t%-30s\t0x%.*" PRIx64 "\n", s->identifier, size, s->value); } static void invalid_type_print(const struct expr *expr) { gmp_printf("0x%Zx [invalid type]", expr->value); } const struct datatype invalid_type = { .type = TYPE_INVALID, .name = "invalid", .print = invalid_type_print, }; static void verdict_type_print(const struct expr *expr) { switch (expr->verdict) { case NF_ACCEPT: printf("accept"); break; case NF_DROP: printf("drop"); break; case NF_QUEUE: printf("queue"); break; case NFT_CONTINUE: printf("continue"); break; case NFT_BREAK: printf("break"); break; case NFT_JUMP: printf("jump %s", expr->chain); break; case NFT_GOTO: printf("goto %s", expr->chain); break; case NFT_RETURN: printf("return"); break; default: BUG(); } } const struct datatype verdict_type = { .type = TYPE_VERDICT, .name = "verdict", .print = verdict_type_print, }; const struct datatype bitmask_type = { .type = TYPE_BITMASK, .name = "bitmask", .basetype = &integer_type, }; static void integer_type_print(const struct expr *expr) { const char *fmt = "%Zu"; if (expr->dtype->basefmt != NULL) fmt = expr->dtype->basefmt; gmp_printf(fmt, expr->value); } static struct error_record *integer_type_parse(const struct expr *sym, struct expr **res) { mpz_t v; mpz_init(v); if (gmp_sscanf(sym->identifier, "%Zu", v) != 1) { mpz_clear(v); if (sym->sym_type != &integer_type) return NULL; return error(&sym->location, "Could not parse %s", sym->sym_type->name); } *res = constant_expr_alloc(&sym->location, sym->sym_type, BYTEORDER_HOST_ENDIAN, 1, NULL); mpz_set((*res)->value, v); mpz_clear(v); return NULL; } const struct datatype integer_type = { .type = TYPE_INTEGER, .name = "integer", .print = integer_type_print, .parse = integer_type_parse, }; static void string_type_print(const struct expr *expr) { unsigned int len = div_round_up(expr->len, BITS_PER_BYTE); char data[len]; mpz_export_data(data, expr->value, BYTEORDER_BIG_ENDIAN, len); printf("\"%s\"", data); } static struct error_record *string_type_parse(const struct expr *sym, struct expr **res) { *res = constant_expr_alloc(&sym->location, &string_type, BYTEORDER_INVALID, (strlen(sym->identifier) + 1) * BITS_PER_BYTE, sym->identifier); return NULL; } const struct datatype string_type = { .type = TYPE_STRING, .name = "string", .print = string_type_print, .parse = string_type_parse, }; static void lladdr_type_print(const struct expr *expr) { unsigned int len = div_round_up(expr->len, BITS_PER_BYTE); const char *delim = ""; uint8_t data[len]; unsigned int i; mpz_export_data(data, expr->value, BYTEORDER_HOST_ENDIAN, len); for (i = 0; i < len; i++) { printf("%s%.2x", delim, data[i]); delim = ":"; } } static struct error_record *lladdr_type_parse(const struct expr *sym, struct expr **res) { char buf[strlen(sym->identifier) + 1], *p; const char *s = sym->identifier; unsigned int len, n; for (len = 0;;) { n = strtoul(s, &p, 16); if (s == p || n > 0xff) return erec_create(EREC_ERROR, &sym->location, "Invalid LL address"); buf[len++] = n; if (*p == '\0') break; s = ++p; } *res = constant_expr_alloc(&sym->location, &lladdr_type, BYTEORDER_HOST_ENDIAN, len * BITS_PER_BYTE, buf); return NULL; } const struct datatype lladdr_type = { .type = TYPE_LLADDR, .name = "LL address", .basetype = &integer_type, .print = lladdr_type_print, .parse = lladdr_type_parse, }; static void ipaddr_type_print(const struct expr *expr) { struct sockaddr_in sin = { .sin_family = AF_INET, }; char buf[NI_MAXHOST]; sin.sin_addr.s_addr = mpz_get_be32(expr->value); getnameinfo((struct sockaddr *)&sin, sizeof(sin), buf, sizeof(buf), NULL, 0, numeric_output ? NI_NUMERICHOST : 0); printf("%s", buf); } static struct error_record *ipaddr_type_parse(const struct expr *sym, struct expr **res) { struct addrinfo *ai, hints = { .ai_family = AF_INET, .ai_socktype = SOCK_DGRAM}; struct in_addr *addr; int err; err = getaddrinfo(sym->identifier, NULL, &hints, &ai); if (err != 0) return error(&sym->location, "Could not resolve hostname: %s", gai_strerror(err)); if (ai->ai_next != NULL) { freeaddrinfo(ai); return error(&sym->location, "Hostname resolves to multiple addresses"); } addr = &((struct sockaddr_in *)ai->ai_addr)->sin_addr; *res = constant_expr_alloc(&sym->location, &ipaddr_type, BYTEORDER_BIG_ENDIAN, sizeof(*addr) * BITS_PER_BYTE, addr); freeaddrinfo(ai); return NULL; } const struct datatype ipaddr_type = { .type = TYPE_IPADDR, .name = "IPv4 address", .basetype = &integer_type, .print = ipaddr_type_print, .parse = ipaddr_type_parse, }; static void ip6addr_type_print(const struct expr *expr) { struct sockaddr_in6 sin6 = { .sin6_family = AF_INET6 }; char buf[NI_MAXHOST]; mpz_export_data(&sin6.sin6_addr, expr->value, BYTEORDER_BIG_ENDIAN, sizeof(sin6.sin6_addr)); getnameinfo((struct sockaddr *)&sin6, sizeof(sin6), buf, sizeof(buf), NULL, 0, numeric_output ? NI_NUMERICHOST : 0); printf("%s", buf); } static struct error_record *ip6addr_type_parse(const struct expr *sym, struct expr **res) { struct addrinfo *ai, hints = { .ai_family = AF_INET6, .ai_socktype = SOCK_DGRAM}; struct in6_addr *addr; int err; err = getaddrinfo(sym->identifier, NULL, &hints, &ai); if (err != 0) return error(&sym->location, "Could not resolve hostname: %s", gai_strerror(err)); if (ai->ai_next != NULL) { freeaddrinfo(ai); return error(&sym->location, "Hostname resolves to multiple addresses"); } addr = &((struct sockaddr_in6 *)ai->ai_addr)->sin6_addr; *res = constant_expr_alloc(&sym->location, &ip6addr_type, BYTEORDER_BIG_ENDIAN, sizeof(*addr) * BITS_PER_BYTE, addr); freeaddrinfo(ai); return NULL; } const struct datatype ip6addr_type = { .type = TYPE_IP6ADDR, .name = "IPv6 address", .basetype = &integer_type, .print = ip6addr_type_print, .parse = ip6addr_type_parse, }; static void inet_protocol_type_print(const struct expr *expr) { struct protoent *p; if (numeric_output < NUMERIC_ALL) { p = getprotobynumber(mpz_get_uint8(expr->value)); if (p != NULL) { printf("%s", p->p_name); return; } } integer_type_print(expr); } static struct error_record *inet_protocol_type_parse(const struct expr *sym, struct expr **res) { struct protoent *p; p = getprotobyname(sym->identifier); if (p == NULL) return error(&sym->location, "Could not resolve protocol name"); *res = constant_expr_alloc(&sym->location, &inet_protocol_type, BYTEORDER_HOST_ENDIAN, BITS_PER_BYTE, &p->p_proto); return NULL; } const struct datatype inet_protocol_type = { .type = TYPE_INET_PROTOCOL, .name = "Internet protocol", .basetype = &integer_type, .print = inet_protocol_type_print, .parse = inet_protocol_type_parse, }; static void inet_service_type_print(const struct expr *expr) { struct sockaddr_in sin = { .sin_family = AF_INET }; char buf[NI_MAXSERV]; sin.sin_port = mpz_get_be16(expr->value); getnameinfo((struct sockaddr *)&sin, sizeof(sin), NULL, 0, buf, sizeof(buf), numeric_output < NUMERIC_ALL ? 0 : NI_NUMERICSERV); printf("%s", buf); } static struct error_record *inet_service_type_parse(const struct expr *sym, struct expr **res) { struct addrinfo *ai; uint16_t port; int err; err = getaddrinfo(NULL, sym->identifier, NULL, &ai); if (err != 0) return error(&sym->location, "Could not resolve service: %s", gai_strerror(err)); port = ((struct sockaddr_in *)ai->ai_addr)->sin_port; *res = constant_expr_alloc(&sym->location, &inet_service_type, BYTEORDER_BIG_ENDIAN, sizeof(port) * BITS_PER_BYTE, &port); freeaddrinfo(ai); return NULL; } const struct datatype inet_service_type = { .type = TYPE_INET_SERVICE, .name = "internet network service", .basetype = &integer_type, .print = inet_service_type_print, .parse = inet_service_type_parse, }; #define RT_SYM_TAB_INITIAL_SIZE 16 struct symbol_table *rt_symbol_table_init(const char *filename) { struct symbolic_constant s; struct symbol_table *tbl; unsigned int size, nelems, val; char buf[512], namebuf[512], *p; FILE *f; size = RT_SYM_TAB_INITIAL_SIZE; tbl = xmalloc(sizeof(*tbl) + size * sizeof(s)); nelems = 0; tbl->size = 4 * BITS_PER_BYTE; tbl->byteorder = BYTEORDER_HOST_ENDIAN; f = fopen(filename, "r"); if (f == NULL) goto out; while (fgets(buf, sizeof(buf), f)) { p = buf; while (*p == ' ' || *p == '\t') p++; if (*p == '#' || *p == '\n' || *p == '\0') continue; if (sscanf(p, "0x%x %511s\n", &val, namebuf) != 2 && sscanf(p, "0x%x %511s #", &val, namebuf) != 2 && sscanf(p, "%u %511s\n", &val, namebuf) != 2 && sscanf(p, "%u %511s #", &val, namebuf) != 2) { fprintf(stderr, "iproute database '%s' corrupted\n", filename); goto out; } /* One element is reserved for list terminator */ if (nelems == size - 2) { size *= 2; tbl = xrealloc(tbl, sizeof(*tbl) + size * sizeof(s)); } tbl->symbols[nelems].identifier = xstrdup(namebuf); tbl->symbols[nelems].value = val; nelems++; } fclose(f); out: tbl->symbols[nelems] = SYMBOL_LIST_END; return tbl; } void rt_symbol_table_free(struct symbol_table *tbl) { const struct symbolic_constant *s; for (s = tbl->symbols; s->identifier != NULL; s++) xfree(s->identifier); xfree(tbl); } static struct symbol_table *mark_tbl; static void __init mark_table_init(void) { mark_tbl = rt_symbol_table_init("/etc/iproute2/rt_marks"); } static void __exit mark_table_exit(void) { rt_symbol_table_free(mark_tbl); } static void mark_type_print(const struct expr *expr) { return symbolic_constant_print(mark_tbl, expr); } static struct error_record *mark_type_parse(const struct expr *sym, struct expr **res) { return symbolic_constant_parse(sym, mark_tbl, res); } const struct datatype mark_type = { .type = TYPE_MARK, .name = "packet mark", .basetype = &integer_type, .basefmt = "0x%.8Zx", .print = mark_type_print, .parse = mark_type_parse, }; static void time_type_print(const struct expr *expr) { uint64_t days, hours, minutes, seconds; const char *delim = ""; seconds = mpz_get_uint64(expr->value); days = seconds / 86400; seconds %= 86400; hours = seconds / 3600; seconds %= 3600; minutes = seconds / 60; seconds %= 60; if (days > 0) { printf("%s%" PRIu64 " d", delim, days); delim = " "; } if (hours > 0) { printf("%s%" PRIu64 " h", delim, hours); delim = " "; } if (minutes > 0) { printf("%s%" PRIu64 " min", delim, minutes); delim = " "; } if (seconds > 0) { printf("%s%" PRIu64 " s", delim, seconds); delim = " "; } } const struct datatype time_type = { .type = TYPE_TIME, .name = "relative time", .basetype = &integer_type, .print = time_type_print, };