Just for testing purposes, I am trying to turn on and off the gpio392 (pin 12 of the J21 header) every 1000 microseconds. However, it is not working. I tried the same example using the sysfs approach and my logic analyzer shows the values correctly. As far as I understand, the gpio392 corresponds to the hardware pin J.00, which according to the TRM has the address 0x02215000, and that is the address I am trying to load in my test. What am I doing wrong?
That is an interesting approach, for sure I will include it in my analysis. What I want to do is to measure the timing overhead of the sysfs approach and the DRA approach in the Jetson TX2 and compare them. Now I guess I will also compare them with the kernel API approach you are suggesting. But to continue I need to solve my DRA problem.
I finally managed to make it work. I deleted the struct to directly modify the registers using offsets. Now I also load the address of the corresponding PADCTL to enable the GPIO:
I couldn’t be a able to calculate address for GPIO_333. How can I find / calculate gpio address for Tx2 ?
My aim is to simulate echo 1 > /sys/class/gpio/gpio333/value with direct registry access way.
I am using tx2 with a custom board. We can open and connect to it with
echo 333 > /sys/class/gpio/export
echo out > /sys/class/gpio/gpio333/direction
echo 1 > /sys/class/gpio/gpio333/value
I need to find the address like you did in your code #define GPIO_392 0x02215000 // J.00 GPIO address
but I am totally lost, for now I am opening /sys/class/gpio/gpio333/value as file and write 1 and 0 to it. But I have speed issues with that.
void* run(void* t)
{
int fd = open(“/dev/mem”, O_RDWR | O_SYNC);
if (fd < 0) {
perror(“/dev/mem”);
fprintf(stderr, “Please run this program as root (for example with sudo)\n”);
exit(1);
}
I am trying to use the code linked by OP (github to read input from the J.00 to J.07 registers as fast as possible by changing the code:
// set up a pointer for convenient access -- this pointer is to the selected GPIO controller
GPIO_mem volatile *pin = (GPIO_mem volatile *)((char *)base + (GPIO_2 & pagemask));
pin->CNF[0] = 0x00ff;
pin->OE[0] = 0xff;
pin->OUT[0] = 0xff;
// pin->IN = 0x00; read only
// disable interrupts
pin->INT_ENB[0] = 0x00;
// don't worry about these for now
//pin->INT_STA[0] = 0x00;
//pin->INT_LVL[0] = 0x000000;
//pin->INT_CLR[0] = 0xffffff;
fprintf(stderr, "press ctrl-C to stop\n");
// "blink" the output values
uint8_t val = 0xff;
while (true) {
sleep(1);
val = val ^ 0xff;
pin->OUT[0] = val;
}
to:
// set up a pointer for convenient access -- this pointer is to the selected GPIO controller
GPIO_mem volatile *pin = (GPIO_mem volatile *)((char *)base + (GPIO_3 & pagemask));
pin->CNF[1] = 0x00ff;
//pin->OE[1] = 0xff;
//pin->OUT[0] = 0xff;
// pin->IN = 0x00; read only
// disable interrupts
//pin->INT_ENB[1] = 0x00;
// don't worry about these for now
//pin->INT_STA[0] = 0x00;
//pin->INT_LVL[0] = 0x000000;
//pin->INT_CLR[0] = 0xffffff;
fprintf(stderr, "press ctrl-C to stop\n");
// "blink" the output values
uint8_t val = 0xff;
int cnt = 0;
while (true) {
printf("%d \n", pin->IN[1]);
}
This code prints 2 ** i when pin J.0i is high. When I set any of pins J.00 to J.03 to high I get the expected output, but when I set any of pins J.04 to J.07 to high I get a seemingly random mix of 0 and the proper value as the output. Does anyone know why this could be? As a final note, pins J.00 to J.03 are I2C in SFIO mode and J.04 to J.07 are I2S in SFIO mode. All 8 pins are set to GPIO mode but this is the only difference I can descern.