печать IP-адресов с использованием GDB - PullRequest
4 голосов
/ 29 ноября 2009

Я отлаживаю сетевой код и хочу напечатать IP-адреса, объявленные как int32. когда я печатаю его с помощью команды gdb print, я получаю некоторые значения, которые не имеют большого значения.

Как я могу распечатать их в значимом формате?

Ответы [ 7 ]

9 голосов
/ 29 ноября 2009

Просто используйте inet_ntoa(3) как:

(gdb) p (char*)inet_ntoa(0x01234567)  # Replace with your IP address
$1 = 0xa000b660 "103.69.35.1"
6 голосов
/ 10 ноября 2012

Если вы отлаживаете файл ядра и не можете использовать inet_ntoa (), вы также можете сделать что-то вроде:

(gdb) set $int_ip = 0x01234567
(gdb) printf "%d.%d.%d.%d\n", ($int_ip & 0xff), ($int_ip >> 8) & 0xff, ($int_ip >> 16) & 0xff, ($int_ip >> 24)
103.69.35.1
(gdb) 
3 голосов
/ 07 декабря 2012

Но inet_ntoa () не принимает аргумент u_int32_t, а скорее аргумент struct in_addr, поэтому предыдущий ответ: p (char *) inet_ntoa (3) мне кажется неправильным.

Вот способ определения функции в файле с именем gdb.cmd.txt, поэтому при запуске вызывайте «gdb -x gdb.cmd.txt».

В файле gdb.cmd.txt введите это:

define ntoa
        set $ipv4 = $arg0
        echo IPV4 =.
        p $ipv4
        set $val1 = ($ipv4 >> 24) & 0xff
        set $val2 = ($ipv4 >> 16) & 0xff
        set $val3 = ($ipv4 >>  8) & 0xff
        set $val4 = ($ipv4 >>  0) & 0xff
       printf "IPV4=%u=0x%02x.%02x.%02x.%02x =%d.%d.%d.%d\n", $ipv4, $val1, $val2, $val3, $val4, $val1, $val2, $val3, $val4
    end

Затем запустите gdb и вызовите ntoa «пользовательскую функцию» следующим образом:

</p> <pre><code>(gdb) ntoa(0x01020304) IPV4 =.$10 = 16909060 IPV4=16909060=0x01.02.03.04 =1.2.3.4 (gdb) ntoa(-1) IPV4 =.$10 = -1 IPV4=4294967295=0xff.ff.ff.ff =255.255.255.255

Кстати: я сейчас ищу, есть ли способ в gdb, чтобы функция возвращала форматированную строку, чтобы я мог запустить команду "p ntoa (0x01020304)" или "p ntoa (ptr-> ipv4_addresss)" ( при условии, что ptr является допустимым ptr для структуры, содержащей элемент данных u_int32_t ipv4_address "). Но, похоже, что пользовательские функции GDB не допускают вызовы sprintf ().

-денис беднар декабрь 2012

1 голос
/ 26 июня 2018

Объяснение ответа Гопината:

(gdb) p/uc (char[4]) 342757386
$4 = {10 '\n', 16 '\020', 110 'n', 20 '\024'}

p/uc: указывает gdb обрабатывать данные как содержимое без знака.

(char[4]) 342757386: переводит IP в массив из 4 символовэлементы, каждый из которых представляет один байт / октет.

Итак, вы говорите gdb обрабатывать десятичное представление IP как массив из четырех байтов - четыре октета - и затем печатать их как беззнаковые символы.

Если вы игнорируете представление ASCII каждого байта, у вас есть ваш IP-адрес: 10.16.110.20.

1 голос
/ 29 ноября 2009

Создайте функцию, которая вызывает inet_ntoa, а затем вызовите ее с помощью команды 'p' в gdb вашего int.

0 голосов
/ 10 ноября 2017

Попробуйте эту команду:

(gdb) p/uc (char[4])342757386
$1 = {10 '\n', 16 '\020', 110 'n', 20 '\024'}
0 голосов
/ 08 декабря 2012

Вот еще один интересный способ:

#include <sys/types.h>  // u_int32_t


// next 3 for inet_ntoa() call
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

// C++ headers
#include <iostream> // std::cout
#include <string>
#include <sstream>  // ostringstream


// class to aid in using the gdb(3) debugger to print a u_int32_t ip_address as a string.
// The toString() is a static method!!
// How to call a C++ method from the gdb debugger, to simulate the inet_nto(3) method
//
// From within gdb debugger, you must have a process, so first stop at main:
//  b main
//  r
//
//  (gdb) p MYIP4::toString(0x0fffefdfc)
//  $1 = "255.254.253.252"
//
//  (gdb) p MYIP4::toString(-1)
//  $2 = "255.255.255.255"
//
//  (gdb) p MYIP4::toString(65536)
//  $3 = "0.1.0.0"
//
// Drawbacks: the a.out executable that you are debugging needs the MYIP4 class already
// compiled and linked into the executable that you are debugging.
//
// PS: I don't know if there is a "slick way" where the MyIP4 class could dynamically be loaded,
// within gdb(), if the executable failed to contain the MYIP4 class.
//
// PS: I had trouble with my operator() idea.. If you know how to improve on it, post away!
//
// @author 1201207 dpb  created
//=============================

class MYIP4
{
public:
    static std::string toString(u_int32_t ipv4_address )
    {
        struct in_addr temp_addr;

        // inet_ntoa() returns a char * to a buffer which may be overwritten
        // soon, so convert char* to a string for extra safety.
        temp_addr.s_addr = htonl(ipv4_address);
        std::string ipv4String = std::string(inet_ntoa( temp_addr ));
        return ipv4String;
    }

#if 0
    // this idea did NOT work, so it is commented out.
    //
    // overload the () operator, so that, within gdb, we can simply type:
    //  p MYIP4(0x01020304)
    // instead of:
    //  p MYIP4::toString(0x01020304)

    std::string operator() ( u_int32_t ipv4_address )
    {
        return toString(ipv4_address);
    }
#endif

};


void test1();

int main()
{
    test1();
    return 0;
}

void test1()
{
    u_int32_t   ipv4Address = 0x01020304;   // host byte order for 1.2.3.4
    std::cout << "Test1: IPAddress=" << MYIP4::toString(ipv4Address) << "\n";
}

...