7
votes

This is a code that gets some info about network the problem is when it prints the MAC address it prints it sometime normally and sometime with fff's like 00:21:84:a2:12:88 and 00:ffffff21:84:a2:12:ffffff88 varies from machine to another

here is the code

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <fcntl.h>
#include <time.h>
#include <sys/ioctl.h>
#include <netinet/in.h>
#include <net/ethernet.h>
#include <net/if.h>
#include <arpa/inet.h>
#include <sys/types.h>
#    include <netinet/if_ether.h>
#define IP_QUAD(ip)  (ip)>>24,((ip)&0x00ff0000)>>16,((ip)&0x0000ff00)>>8,((ip)&0x000000ff)
#define IP_ADDR_LEN 4
struct {
    char *dev;
    int sd;
    struct ether_addr eth;
    struct in_addr ip;
    struct in_addr bcast;
    unsigned int mtu;
} local_info ;

struct ifreq eth_init(char*,struct ifreq);
struct ifreq eth_get_info(struct ifreq);
struct ifreq eth_get_bcast(struct ifreq);

int
main(int argc,char **argv){
    int sd;
    struct ifreq ifr;
    if(argc != 2){
    fprintf(stderr,"usage: <command> <devicename>\n");
    exit(1);
    }
    ifr = eth_init(argv[1],ifr);
    ifr = eth_get_info(ifr);

    printf("> Exiting...\n");
    return(0);
}

struct ifreq
eth_init(char *dev,struct ifreq ifr){
//Intitating Socket     
if((local_info.sd = socket(PF_INET,SOCK_PACKET,(ETH_P_ALL))) < 0){
        printf("> Error initating the ethernet socket..\n");
        exit(-1);
    }
//Yupeeeeeeee Descriptor open 
    printf("> Initated Ethernet socket on Descriptor (%x)\n",local_info.sd);
//Set global variables
    local_info.dev = dev;
    return ifr;
}
struct ifreq
eth_get_info(struct ifreq ifr){
    int i = ETHER_ADDR_LEN;
    char* ptr;
    memset(&ifr,0,sizeof(ifr));
    strncpy(ifr.ifr_name,local_info.dev,sizeof(ifr.ifr_name));
    //Getting MAC
    if(ioctl(local_info.sd,SIOCGIFHWADDR,&ifr) < 0){
            printf("> Error Getting the Local Mac address\n");
            exit(-1);
        }
    printf("> Successfully received Local MAC Address : %02x:%02x:%02x:%02x:%02x:%02x\n",
                ifr.ifr_hwaddr.sa_data[0],ifr.ifr_hwaddr.sa_data[1],ifr.ifr_hwaddr.sa_data[2]
                ,ifr.ifr_hwaddr.sa_data[3],ifr.ifr_hwaddr.sa_data[4],ifr.ifr_hwaddr.sa_data[5]);

        memcpy(&(local_info.eth),&ifr.ifr_hwaddr.sa_data,ETH_ALEN);


    // Getting IP Address
        memset(&ifr,0,sizeof(ifr));
        strncpy(ifr.ifr_name,local_info.dev,sizeof(ifr.ifr_name));
        if( ioctl(local_info.sd,SIOCGIFADDR,&ifr) < 0){ 
                printf("> Error gettint the local IP address\n");
                exit(-1);
        }   
        printf("> Successfully received the IP Address %s\n",inet_ntoa((*(struct sockaddr_in *)&ifr.ifr_addr).sin_addr));
        memcpy(&(local_info.ip.s_addr),&(*(struct sockaddr_in *)&ifr.ifr_addr).sin_addr.s_addr,IP_ADDR_LEN);

            // Get MTU
        memset(&ifr,0,sizeof(ifr));
        strncpy(ifr.ifr_name,local_info.dev,sizeof(ifr.ifr_name));
        if ( ioctl(local_info.sd,SIOCGIFMTU,&ifr) < 0){
                printf("> Error Getting the MTU Value\n");
                exit(-1);
        }
        printf("> Recevied Successfully the MTU Value \n");
        local_info.mtu = ifr.ifr_mtu;


    return ifr;
}

struct ifreq
eth_get_bcast(struct ifreq ifr){
/* get broadcast addr for size */
        memset(&ifr,0,sizeof(ifr));
        strncpy(ifr.ifr_name, local_info.dev, sizeof (ifr.ifr_name));
        if (ioctl(local_info.sd, SIOCGIFBRDADDR, &ifr) < 0 ) { 
           printf("> Error getting the Broadcast address\n");
           exit(-1);
        }
        printf("> Received the BroadCast address: %s\n",inet_ntoa((*(struct sockaddr_in *)&ifr.ifr_addr).sin_addr));
        memcpy(&(local_info.bcast.s_addr),
               &(*(struct sockaddr_in *)&ifr.ifr_addr).sin_addr.s_addr,
               IP_ADDR_LEN);

    return ifr;
}

the problem is the the eth_get_info function the MAC sector , the printing statement

any solutions how to fix that?

4
tip: don't use <pre> to highlight code here. Just select the whole code block and click on the {} button in the editor.Mat
ifr.ifr_hwaddr.sa_data[0], <= these are probably chars, if you bit mask it down to the low byte, it'll probably work: (int)(ifr.ifr_hwaddr.sa_data[0] & 0xff) for each inputforsvarir
@forsvarir... I don't get what u mean!Saad Talaat
In your print statement, you're passing characters to a print format (%02X), that's expecting an integer value, so the value of the top 3 bytes is undefined, hence the FFs you're seeing. If you and the arguments, it removes the top three bytes and you should get the right value: (int)(ifr.ifr_hwaddr.sa_data[0] & 0xff)forsvarir

4 Answers

14
votes

It looks like a signed/unsigned problem.

Try to cast into unsigned char :

  printf("> Successfully received Local MAC Address : %02x:%02x:%02x:%02x:%02x:%02x\n",
  (unsigned char) ifr.ifr_hwaddr.sa_data[0],
  (unsigned char) ifr.ifr_hwaddr.sa_data[1],
  (unsigned char) ifr.ifr_hwaddr.sa_data[2],
  (unsigned char) ifr.ifr_hwaddr.sa_data[3],
  (unsigned char) ifr.ifr_hwaddr.sa_data[4],
  (unsigned char) ifr.ifr_hwaddr.sa_data[5]);
8
votes

Although there is already an accepted answer here, there is a better solution in netinet/ether.h.

Given that Mac addresses are typically stored in u_int8_t types, as in the ether_addr struct:

You can simply do this:

printf("Mac Address: %s", ether_ntoa((struct ether_addr*)ar->sa));

in my Case ar is something that looks like this:

struct {
   u_int8_t sa[6];
}

You can easily copy this into another buffer with something like asprintf:

char *formatted_mac_address;
asprintf(formatted_mac_address, "Mac Address: %s", ether_ntoa((struct ether_addr*)ar->sa));

If you don't have a struct as I do, you can also just use the address of any u_int8_t in place of ar->sa.

Appropriate headers/etc should be pulled in, but that's going to look a lot neater than the accepted solution here.

0
votes

I prefer use an explicit length modifier in format string for values shorter than int. For example %02hhx instead of %02x for one-byte values. It allow me to not worry about those subtle conversion and promotion issues:

#include <stdio.h>

int main(void)
{
    signed char sff = '\xff';
    unsigned char uff = '\xff';

    printf("signed:   %02x %02hhx\n", sff, sff);
    printf("unsigned: %02x %02hhx\n", uff, uff);

    return 0;
}

prints

signed:   ffffffff ff
unsigned: ff ff
-2
votes

Here is a simple C program to find the MAC address of system:

#include<stdio.h>
#include<stdlib.h>

int main() {
    system("getmac");
    return 0;
}