将unsigned char数组插入数据库的问题

时间:2013-05-06 15:03:39

标签: mysql c linux types

这是我的代码。它应该做254个ARP请求并将ip和mac地址存储到unsigned char数组中,之后将它们存储到数据库中。除数据库部分外,这一切似乎都很完美。它我只正确存储第一个值(计数器值)但不存储数组。我尝试将它们作为%u,%X传递但是这些格式期望无符号整数但没有未签名的char数组。任何帮助将受到高度赞赏。

#include <stdio.h>
#include <string.h>
#include <sys/ioctl.h>
#include <net/if.h>
#include <sys/socket.h>
#include <netpacket/packet.h>
#include <net/ethernet.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <fcntl.h>
#include <sys/time.h>
#include <sys/stat.h>
#include <mysql.h>



unsigned char macOrigen[6];
unsigned char ipOrigen[4];
unsigned char ipDestino[4];
unsigned char tramaEnviar[1514];
unsigned char tramaRecibir[1514];
unsigned char mascaraSubred[4];
unsigned char ipDestino[4];
unsigned char respuestaArp[2] = {0x00, 0x02};
unsigned char protocolo[2] = {0x08, 0x00};
unsigned char ethertype[2] = {0x08, 0x06};
unsigned char macDestino[6] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
unsigned char macBroadcast[6] = {0XFF, 0XFF, 0XFF, 0XFF, 0XFF, 0XFF};
unsigned char hardware[1] = {0x01};
unsigned char longitudHW[1] = {0x06};
unsigned char longitudIP[1] = {0x04};
unsigned char codigoOperacionSolicitud[2] = {0x00, 0x01};

int ObtenerDatos (int packetSocket){
    char nombre[10];
    int index;
    struct ifreq interfaz;
    printf("Ingrese el nombre de la interfaz:\nPuede consultar las inerfaces disponibles tecleando 'ifconfig' en otra terminal.\n\n");
    gets(nombre);
    strcpy(interfaz.ifr_name, nombre);
    if(ioctl(packetSocket, SIOCGIFINDEX, &interfaz) == -1){
        perror("Error al obener el índice.\n");
        }else{
            index = interfaz.ifr_ifindex;
        }
    if(ioctl(packetSocket, SIOCGIFHWADDR, &interfaz) == -1)
    {
        perror("No se ha podido obtener la informacion del dispositivo.\n");
    }else{
        memcpy(macOrigen, interfaz.ifr_hwaddr.sa_data, 6);      
    }
    if(ioctl(packetSocket, SIOCGIFADDR, &interfaz) == -1){
        perror("No se ha podido obtener la informacion del dispositivo.\n");
    }else{
        memcpy(ipOrigen, interfaz.ifr_addr.sa_data+2, 4);
    }if(ioctl(packetSocket, SIOCGIFNETMASK, &interfaz) == -1){
        perror("Error al obtener la mascara de subred.\n");
    }else{
        memcpy(mascaraSubred, interfaz.ifr_netmask.sa_data+2, 4);
    }

    return index;
}

void estructuraTramaArp(unsigned char *tramaEnviar){
    memcpy(tramaEnviar, macBroadcast, 6);
    memcpy(tramaEnviar+6, macOrigen, 6);
    memcpy(tramaEnviar+12, ethertype, 2);
    memcpy(tramaEnviar+14, hardware, 1);
    memcpy(tramaEnviar+15, protocolo, 2);
    memcpy(tramaEnviar+17, longitudHW, 1);
    memcpy(tramaEnviar+18, longitudIP, 1);
    memcpy(tramaEnviar+19, longitudIP, 1);
    memcpy(tramaEnviar+20, codigoOperacionSolicitud, 2);
    memcpy(tramaEnviar+22, macOrigen, 6);
    memcpy(tramaEnviar+28, ipOrigen, 4);
    memcpy(tramaEnviar+32, macBroadcast, 6);
    memcpy(tramaEnviar+38, ipDestino, 4);
}

void solicitudArp(int packetSocket, unsigned char *tramaEnviar, int index){
    int tam;
    struct sockaddr_ll nic;
    memset(&nic, 0x00, sizeof(nic));
    nic.sll_family = AF_PACKET;
    nic.sll_protocol = htons(ETH_P_ALL);
    nic.sll_ifindex = index;
    tam = sendto(packetSocket, tramaEnviar, 60, 0, (struct sockaddr *)&nic, sizeof(nic));
    if (tam == -1){
        perror("Error al enviar la solicitud ARP");

    }else("Exito al enviar la trama");
}

void recibeTramasArp(int packetSocket, unsigned char *tramaRecibir, int longitud){

    struct timeval tiempoEspera;
    tiempoEspera.tv_sec = 0;
    tiempoEspera.tv_usec = 900000L;
    int tam, res, i, j;
    fd_set lectura, escritura; 
    while(1)
    {
        FD_ZERO(&lectura);
        FD_SET(packetSocket, &escritura);
        if((res=select(packetSocket+1, &lectura, NULL, NULL, &tiempoEspera)) > 0)
        {
            if(FD_ISSET(packetSocket, &lectura)){
                if((tam = recvfrom(packetSocket, tramaRecibir, longitud, 0, NULL, 0)) < 0){
                    perror("Error al recibir.\n");
                    break;
                    }else{
                        if(!memcmp(tramaRecibir+0, macOrigen, 6) && !memcmp(tramaRecibir+12,ethertype, 2) && !memcmp(tramaRecibir+16,protocolo,2) && !memcmp(tramaRecibir+20,respuestaArp,2) && !memcmp(tramaRecibir+28,ipDestino,4) && !memcmp(tramaRecibir+38,ipOrigen,4) ){
        printf("La dirección IP:\n");
        for(i = 28; i <= 31; i++){
        printf("%.2d:",tramaRecibir[i]);
        }
        printf("\nRespondio con la direccion mac:\n");
        for(j = 22; j <=27; j++){
            printf("%.2x.",tramaRecibir[j]);
        }
        printf("\n");
        break;

}
    printf("No se recibio respuesta.\n");
}
}
    break;
}else{
    printf("\nEl tiempo de espera se agoto.\n\n");
    break;  
}

}

}

int main(int argc, char **argv){
    int indice, i, j; 
    char stmt_buf[100];
    char bd [100];
    int packetSocket = socket(AF_PACKET, SOCK_RAW, htons(ETH_P_ALL));
    MYSQL *con = mysql_init(NULL);
    if (con == NULL){
        fprintf(stderr,"%s\n", mysql_error(con));

    }
    if (mysql_real_connect(con, "localhost", "angel", "xmdapoe", "arp", 0, NULL, 0) == NULL){
        fprintf(stderr,"%s\n", mysql_error(con));
    }
    if (mysql_query(con, "DROP TABLE IF EXISTS info"))
    {
        fprintf(stderr, "%s\n", mysql_error(con));
    }   
    if (mysql_query(con, "CREATE TABLE info(id INT, ip TEXT, mac TEXT)"))
    {
        fprintf(stderr, "%s\n", mysql_error(con));
    }
    if (packetSocket == -1)
    {
        perror("Error al abrir el socket.\n");
    }else{
        indice = ObtenerDatos(packetSocket);
        memcpy(ipDestino, ipOrigen, 4);
            ipDestino[3] = 1;
        for(i = 0; i<254 ; i++){
            printf("Direccion IP destino:");
            printf("\n");
            for (j = 0; j<4; j++){
                printf("%d.",ipDestino[j]);
                }   
                estructuraTramaArp(tramaEnviar);
                solicitudArp(packetSocket, tramaEnviar, indice);
                recibeTramasArp(packetSocket, tramaRecibir, 1514);
                sprintf(stmt_buf, "insert into info values ('%d', '%.2x', '%.2x')", i, ipOrigen,macOrigen);
                if(mysql_query(con, stmt_buf)){
                    fprintf(stderr, "%s\n",mysql_error(con));               
                }
                ipDestino[3]++;
            }
            }
    close(packetSocket);
}

2 个答案:

答案 0 :(得分:0)

使用%。2x的sprintf你只需要打印ipOrigen和macOrigen的内存地址。 我认为你想要的东西可以通过以下方式实现:

    sprintf (buf, "%d%d%d%d", ipOrigen[0], ipOrigen[1], ipOrigen[2], ipOrigen[3]);

答案 1 :(得分:0)

查看你的sprintf语句,你实际上是在打印ipOrigen和macOrigen数组的第一个元素的地址。对于ipOrigen,您可以执行以下操作:

printf("%x", htonl(*(uint32_t *)ipOrigen));

对于macOrigen,您可以使用endian.h中的htobe64()来使用类似的方法。