Очередь сетевого фильтра Iptables nfq_set_verdict_mark (nfq_set_verdict2), похоже, не применяет метку

У меня проблемы с iptables, и свежий взгляд, вероятно, сможет увидеть, в чем проблема.

Я пытаюсь использовать очередь netfilter для маркировки пакетов (через nfq_set_verdict_mark — в старой документации говорится об использовании nfq_set_verdict2, но этой функции, похоже, больше нет в заголовке или библиотеке) в цепочке PREROUTING таблицы mangle. Затем пакет должен перейти к цепочке PREROUTING таблицы nat, где у меня есть правила для чтения метки и соответствующего перенаправления. Действительно, это работало в более раннем тесте, но теперь нет!

Вот функция обратного вызова для моего упрощенного теста — она просто помечает 0xFFFFFFFF и принимает:

static int cb(struct nfq_q_handle *qh, struct nfgenmsg *nfmsg,
          struct nfq_data *nfa, void *data)
{
    uint32_t id = print_pkt(nfa); // from example code
    printf("marking packet...\n");
    return nfq_set_verdict_mark(qh,id,NF_ACCEPT,0xFFFFFFFF,0,NULL);
}

Вот правила iptable, которые я использую:

// Using queue -- not working
iptables -t mangle -A PREROUTING -p tcp --dport 80 -j QUEUE
iptables -t nat -A PREROUTING -p tcp --dport 80 -m mark --mark 0xFFFFFFFF -j REDIRECT --to-ports 9009

Я также использую conntrack --flush для сброса состояния соединения, чтобы гарантировать, что пакеты попадут в таблицу nat. Я использую 0xFFFFFFFF, чтобы предотвратить возможные проблемы с порядком байтов. Пакеты четко попадают в очередь и принимаются ("отметка пакета..." печатается, браузер может получить доступ к содержимому на порту 80). Однако перенаправления не происходит.

Если я ставлю метку напрямую, а не в очередь, перенаправление работает. Как будто метка не добавляется в очередь. Это, конечно, маркировалось раньше, и я не менял версии библиотек или что-то в этом роде.

// Without queue -- redirection happens
iptables -t mangle -A PREROUTING -p tcp --dport 80 -j MARK --set-mark 0xFFFFFFFF
iptables -t nat -A PREROUTING -p tcp --dport 80 -m mark --mark 0xFFFFFFFF -j REDIRECT --to-ports 9009

Вот print_pkt на всякий случай - написан не мной, а взят из примера netfilter_queue, не менялся и раньше работал:

static u_int32_t print_pkt (struct nfq_data *tb)
{
        int id = 0;
        struct nfqnl_msg_packet_hdr *ph;
        struct nfqnl_msg_packet_hw *hwph;
        u_int32_t mark,ifi; 
        int ret;
        char *data;

        ph = nfq_get_msg_packet_hdr(tb);
        if (ph) {
                id = ntohl(ph->packet_id);
                printf("hw_protocol=0x%04x hook=%u id=%u ",
                        ntohs(ph->hw_protocol), ph->hook, id);
        }

        hwph = nfq_get_packet_hw(tb);
        if (hwph) {
                int i, hlen = ntohs(hwph->hw_addrlen);

                printf("hw_src_addr=");
                for (i = 0; i < hlen-1; i++)
                        printf("%02x:", hwph->hw_addr[i]);
                printf("%02x ", hwph->hw_addr[hlen-1]);
        }

        mark = nfq_get_nfmark(tb);
        if (mark)
                printf("mark=%u ", mark);

        ifi = nfq_get_indev(tb);
        if (ifi)
                printf("indev=%u ", ifi);

        ifi = nfq_get_outdev(tb);
        if (ifi)
                printf("outdev=%u ", ifi);
        ifi = nfq_get_physindev(tb);
        if (ifi)
                printf("physindev=%u ", ifi);

        ifi = nfq_get_physoutdev(tb);
        if (ifi)
                printf("physoutdev=%u ", ifi);

        ret = nfq_get_payload(tb, &data);
        if (ret >= 0)
                printf("payload_len=%d ", ret);

        fputc('\n', stdout);

        return id;
}

Также основная функция (тоже взята из кода примера и ранее работала):

int main(int argc, char **argv)
{
        struct nfq_handle *h;
        struct nfq_q_handle *qh;
        struct nfnl_handle *nh;
        int fd;
        int rv;
        char buf[4096] __attribute__ ((aligned));

        printf("opening library handle\n");
        h = nfq_open();
        if (!h) {
                fprintf(stderr, "error during nfq_open()\n");
                exit(1);
        }

        printf("unbinding existing nf_queue handler for AF_INET (if any)\n");
        if (nfq_unbind_pf(h, AF_INET) < 0) {
                fprintf(stderr, "error during nfq_unbind_pf()\n");
                exit(1);
        }

        printf("binding nfnetlink_queue as nf_queue handler for AF_INET\n");
        if (nfq_bind_pf(h, AF_INET) < 0) {
                fprintf(stderr, "error during nfq_bind_pf()\n");
                exit(1);
        }

        printf("binding this socket to queue '0'\n");
        qh = nfq_create_queue(h,  0, &cb, NULL);
        if (!qh) {
                fprintf(stderr, "error during nfq_create_queue()\n");
                exit(1);
        }

        printf("setting copy_packet mode\n");
        if (nfq_set_mode(qh, NFQNL_COPY_META, 0xffff) < 0) {
                fprintf(stderr, "can't set packet_copy mode\n");
                exit(1);
        }

        fd = nfq_fd(h);

        while ((rv = recv(fd, buf, sizeof(buf), 0)) && rv >= 0) {
                printf("pkt received\n");
                nfq_handle_packet(h, buf, rv);
        }

        printf("unbinding from queue 0\n");
        nfq_destroy_queue(qh);

#ifdef INSANE
        /* normally, applications SHOULD NOT issue this command, since
         * it detaches other programs/sockets from AF_INET, too ! */
        printf("unbinding from AF_INET\n");
        nfq_unbind_pf(h, AF_INET);
#endif

        printf("closing library handle\n");
        nfq_close(h);

        exit(0);
}

Перенаправление — это ожидаемое поведение в моем тестовом коде, но этого не происходит. Какое недостающее звено? Я потратил часы на эту проблему (которая работала, как и ожидалось) безрезультатно. Помощь!!


person 1337 on Tuesdays    schedule 18.12.2013    source источник


Ответы (2)


старая документация говорит об использовании nfq_set_verdict2, но этой функции, похоже, больше нет в заголовке или библиотеке

Возможно, вы используете устаревшую версию libnetfilter_queue hdrs+lib? Версия 1.0.2 (netfilter.org), по-видимому, является последним выпуском и имеет эту функцию.

person Michael Joost    schedule 08.02.2014

Да, с последней версией libnetfilter_queue следует использовать только функцию nfq_set_verdict2, как показано ниже.

 nfq_set_verdict2(handle,
                  packet_id,
                  NF_AXXEPT,
                  nfq_get_nfmark(nfa) | 0xFFFFFFFF,
                  0,
                  NULL);

Это сработало для меня.

person Nilesh Awate    schedule 02.03.2021