How to print a pointer address in hex?

Question about which tools to use, bugs, the best way to implement a function, etc should go here. Don't forget to see if your question is answered in the wiki first! When in doubt post here.
Post Reply
AwfulMint
Member
Member
Posts: 35
Joined: Wed Nov 01, 2017 7:41 pm

How to print a pointer address in hex?

Post by AwfulMint »

I'm starting in os dev and I have in my kernel, functions to print to the terminal, cursor update and serial communication. I'm searching for a long time and nothing :/

I want to know how can I print the address of a pointer to screen and get this address and convert it to a string and send it to my serial port. Plz help me dudes, I'm dumb #-o
OS Development is awesome!

|AetherOS Project|
User avatar
justinian
Posts: 19
Joined: Mon Apr 23, 2018 6:25 pm
Location: San Francisco

Re: How to print a pointer address in hex?

Post by justinian »

Well, first you need to cast the pointer to an appropriately-sized unsigned integer. (uint32_t, or uint64_t on a 64bit cpu in long mode.) Now, every 4 bits of that integer is one hex digit. So to convert it to a string, you can just shift and mask to get just the 4 bits you want, and map that into a static global array of characters ("0123456789abcdef"). To get the number in the right readable direction, make sure you're converting the most significant bits first.

Here's an very simple example from my bootloader.
Developing: jsix - UEFI-booted x64 kernel
AwfulMint
Member
Member
Posts: 35
Joined: Wed Nov 01, 2017 7:41 pm

Re: How to print a pointer address in hex?

Post by AwfulMint »

Ok, I'll try to adapt it to my needs. When casting a common pointer to a uint32_t pointer for example, I can get the common pointer memory address?

I'm searchin ways to debug my kernel, I'm using serial ports bcs I don't know how to make my own printf function.
OS Development is awesome!

|AetherOS Project|
AwfulMint
Member
Member
Posts: 35
Joined: Wed Nov 01, 2017 7:41 pm

Re: How to print a pointer address in hex?

Post by AwfulMint »

So I really get strange things, first when trying to convert a pointer to uint32_t pointer not seems to work.

Second thing, what this means: 0000694f00000000

Putting it into an Hex to ASCII I get crazy symbols and iO (yes inverted).

My code:

Code: Select all

const uint16_t digits[] = {
    u'0', u'1', u'2', u'3', u'4', u'5', u'6', u'7', u'8', u'9', u'a', u'b', u'c', u'd', u'e', u'f'
};

void printHexDigit(uint32_t n) {
    char buffer[9];
    char *p = buffer;

    for (int i = 7; i >= 0; --i) {
	uint8_t nibble = (n & (0xf << (i*4))) >> (i*4);
	*p++ = digits[nibble];
    }
    *p = 0;
    writeStr(SERIAL_COM1, buffer);
}

void kmain(void) {
    Console *console = NULL;
    
    ttyInitialize(console);
    
    writeStr(SERIAL_COM1, "Iniciando log...\n");
    writeStr(SERIAL_COM1, "Aether Kernel v0.01 BUILD 0001, estágio 1 iniciado!\n\n");
    
    uint8_t c[] = {'O', 'i'};
    
    uint32_t *p = (uint32_t*) c;
    
    printHexDigit(p[0]);
    printHexDigit(p[1]);
    
    println(console, "Aether Kernel v0.01 BUILD 0001");
}
OS Development is awesome!

|AetherOS Project|
AwfulMint
Member
Member
Posts: 35
Joined: Wed Nov 01, 2017 7:41 pm

Re: How to print a pointer address in hex?

Post by AwfulMint »

Ow there's one more thing, this function convert a string to hex. I want to get the memory location of a pointer and prints it as hex example:

Code: Select all

SomeType *p = ...;

writeHex(*p);
and the out something like: 0xA9924354
OS Development is awesome!

|AetherOS Project|
User avatar
justinian
Posts: 19
Joined: Mon Apr 23, 2018 6:25 pm
Location: San Francisco

Re: How to print a pointer address in hex?

Post by justinian »

AwfulMint wrote:

Code: Select all

const uint16_t digits[] = {
    u'0', u'1', u'2', u'3', u'4', u'5', u'6', u'7', u'8', u'9', u'a', u'b', u'c', u'd', u'e', u'f'
};
You probably don't want uint16_t for your implementation. In this case I was working with UEFI, which uses wide characters. Unless you're also in UEFI-land, you most likely want just char. (Which also means you don't need the u prefix on your character literals.
AwfulMint wrote:

Code: Select all

    uint8_t c[] = {'O', 'i'};
    
    uint32_t *p = (uint32_t*) c;
    
    printHexDigit(p[0]);
    printHexDigit(p[1]);
Take a step back and think about what you're asking the compiler for with the data types here. Remember than in C, a pointer and an array are not very different. So you're telling the compiler to allocate two bytes on the stack for 'Oi', and c points to those two bytes. You then make p point to the same address, and then ask it to read 4 bytes starting at p as a uint32_t, and then the next 4 bytes after that as a uint32_t, when you only allocated 2 bytes.

What you probably want here is the address of each uint8_t in array c. And you want to convert the address (not the value) to a uint32_t. So instead of casting it as pointer (uint32_t*) you just want to treat it as a value (uint32_t).

You might want to refresh yourself on pointers and datatypes (here's a short tutorial). As for why you see "iO" in the output, it's because you're treating that memory as a 32-bit number on a little-endian computer - what used to be known as the "NUXI" problem on old UNIX machines.
Developing: jsix - UEFI-booted x64 kernel
Post Reply