|
@@ -130,13 +130,16 @@ struct wiringPiNodeStruct *wiringPiNodes = NULL ; |
|
|
// Access from ARM Running Linux |
|
|
// Access from ARM Running Linux |
|
|
// Taken from Gert/Doms code. Some of this is not in the manual |
|
|
// Taken from Gert/Doms code. Some of this is not in the manual |
|
|
// that I can find )-: |
|
|
// that I can find )-: |
|
|
|
|
|
// |
|
|
|
|
|
// Updates in September 2015 - all now static variables (and apologies for the caps) |
|
|
|
|
|
// due to the Pi v2 and the new /dev/gpiomem interface |
|
|
|
|
|
|
|
|
static volatile unsigned int BCM2708_PERI_BASE = 0x20000000 ; // Variable for Pi2 |
|
|
|
|
|
#define GPIO_PADS (BCM2708_PERI_BASE + 0x00100000) |
|
|
|
|
|
#define CLOCK_BASE (BCM2708_PERI_BASE + 0x00101000) |
|
|
|
|
|
#define GPIO_BASE (BCM2708_PERI_BASE + 0x00200000) |
|
|
|
|
|
#define GPIO_TIMER (BCM2708_PERI_BASE + 0x0000B000) |
|
|
|
|
|
#define GPIO_PWM (BCM2708_PERI_BASE + 0x0020C000) |
|
|
|
|
|
|
|
|
static volatile unsigned int RASPBERRY_PI_PERI_BASE ; |
|
|
|
|
|
static volatile unsigned int GPIO_PADS ; |
|
|
|
|
|
static volatile unsigned int GPIO_CLOCK_BASE ; |
|
|
|
|
|
static volatile unsigned int GPIO_BASE ; |
|
|
|
|
|
static volatile unsigned int GPIO_TIMER ; |
|
|
|
|
|
static volatile unsigned int GPIO_PWM ; |
|
|
|
|
|
|
|
|
#define PAGE_SIZE (4*1024) |
|
|
#define PAGE_SIZE (4*1024) |
|
|
#define BLOCK_SIZE (4*1024) |
|
|
#define BLOCK_SIZE (4*1024) |
|
@@ -628,6 +631,7 @@ int wiringPiFailure (int fatal, const char *message, ...) |
|
|
* 0011 - Pi CM, Rev 1.2, 512MB, Sony |
|
|
* 0011 - Pi CM, Rev 1.2, 512MB, Sony |
|
|
* 0012 - Model A+ Rev 1.2, 256MB, Sony |
|
|
* 0012 - Model A+ Rev 1.2, 256MB, Sony |
|
|
* 0014 - Pi CM, Rev 1.1, 512MB, Sony (Actual Revision might be different) |
|
|
* 0014 - Pi CM, Rev 1.1, 512MB, Sony (Actual Revision might be different) |
|
|
|
|
|
* 0015 - Model A+ Rev 1.1, 256MB, Sony |
|
|
* |
|
|
* |
|
|
* For the Pi 2: |
|
|
* For the Pi 2: |
|
|
* 0010 - Model 2, Rev 1.1, Quad Core, 1GB, Sony |
|
|
* 0010 - Model 2, Rev 1.1, Quad Core, 1GB, Sony |
|
@@ -685,7 +689,11 @@ int piBoardRev (void) |
|
|
else if (strstr (line, "BCM2708") == NULL) |
|
|
else if (strstr (line, "BCM2708") == NULL) |
|
|
{ |
|
|
{ |
|
|
fprintf (stderr, "Unable to determine hardware version. I see: %s,\n", line) ; |
|
|
fprintf (stderr, "Unable to determine hardware version. I see: %s,\n", line) ; |
|
|
fprintf (stderr, " - expecting BCM2708 or BCM2709. Please report this to projects@drogon.net\n") ; |
|
|
|
|
|
|
|
|
fprintf (stderr, " - expecting BCM2708 or BCM2709.\n") ; |
|
|
|
|
|
fprintf (stderr, "If this is a genuine Raspberry Pi then please report this\n") ; |
|
|
|
|
|
fprintf (stderr, "to projects@drogon.net. If this is not a Raspberry Pi then you\n") ; |
|
|
|
|
|
fprintf (stderr, "are on your own as wiringPi is designed to support the\n") ; |
|
|
|
|
|
fprintf (stderr, "Raspberry Pi ONLY.\n") ; |
|
|
exit (EXIT_FAILURE) ; |
|
|
exit (EXIT_FAILURE) ; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
@@ -860,6 +868,7 @@ void piBoardId (int *model, int *rev, int *mem, int *maker, int *overVolted) |
|
|
else if (strcmp (c, "0012") == 0) { *model = PI_MODEL_AP ; *rev = PI_VERSION_1_2 ; *mem = 256 ; *maker = PI_MAKER_SONY ; } |
|
|
else if (strcmp (c, "0012") == 0) { *model = PI_MODEL_AP ; *rev = PI_VERSION_1_2 ; *mem = 256 ; *maker = PI_MAKER_SONY ; } |
|
|
else if (strcmp (c, "0013") == 0) { *model = PI_MODEL_BP ; *rev = PI_VERSION_1_2 ; *mem = 512 ; *maker = PI_MAKER_MBEST ; } |
|
|
else if (strcmp (c, "0013") == 0) { *model = PI_MODEL_BP ; *rev = PI_VERSION_1_2 ; *mem = 512 ; *maker = PI_MAKER_MBEST ; } |
|
|
else if (strcmp (c, "0014") == 0) { *model = PI_MODEL_CM ; *rev = PI_VERSION_1_2 ; *mem = 512 ; *maker = PI_MAKER_SONY ; } |
|
|
else if (strcmp (c, "0014") == 0) { *model = PI_MODEL_CM ; *rev = PI_VERSION_1_2 ; *mem = 512 ; *maker = PI_MAKER_SONY ; } |
|
|
|
|
|
else if (strcmp (c, "0015") == 0) { *model = PI_MODEL_AP ; *rev = PI_VERSION_1_1 ; *mem = 256 ; *maker = PI_MAKER_SONY ; } |
|
|
else { *model = 0 ; *rev = 0 ; *mem = 0 ; *maker = 0 ; } |
|
|
else { *model = 0 ; *rev = 0 ; *mem = 0 ; *maker = 0 ; } |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
@@ -1829,9 +1838,6 @@ int wiringPiSetup (void) |
|
|
if (getenv (ENV_CODES) != NULL) |
|
|
if (getenv (ENV_CODES) != NULL) |
|
|
wiringPiReturnCodes = TRUE ; |
|
|
wiringPiReturnCodes = TRUE ; |
|
|
|
|
|
|
|
|
if (geteuid () != 0) |
|
|
|
|
|
(void)wiringPiFailure (WPI_FATAL, "wiringPiSetup: Must be root. (Did you forget sudo?)\n") ; |
|
|
|
|
|
|
|
|
|
|
|
if (wiringPiDebug) |
|
|
if (wiringPiDebug) |
|
|
printf ("wiringPi: wiringPiSetup called\n") ; |
|
|
printf ("wiringPi: wiringPiSetup called\n") ; |
|
|
|
|
|
|
|
@@ -1844,43 +1850,73 @@ int wiringPiSetup (void) |
|
|
} |
|
|
} |
|
|
else // A, B, Rev 2, B+, CM, Pi2 |
|
|
else // A, B, Rev 2, B+, CM, Pi2 |
|
|
{ |
|
|
{ |
|
|
if (piModel2) |
|
|
|
|
|
BCM2708_PERI_BASE = 0x3F000000 ; |
|
|
|
|
|
pinToGpio = pinToGpioR2 ; |
|
|
pinToGpio = pinToGpioR2 ; |
|
|
physToGpio = physToGpioR2 ; |
|
|
physToGpio = physToGpioR2 ; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
// Open the master /dev/memory device |
|
|
|
|
|
|
|
|
if (piModel2) |
|
|
|
|
|
RASPBERRY_PI_PERI_BASE = 0x3F000000 ; |
|
|
|
|
|
else |
|
|
|
|
|
RASPBERRY_PI_PERI_BASE = 0x20000000 ; |
|
|
|
|
|
|
|
|
|
|
|
// Open the master /dev/ memory control device |
|
|
|
|
|
|
|
|
|
|
|
// See if /dev/gpiomem exists and we can open it... |
|
|
|
|
|
|
|
|
|
|
|
if ((fd = open ("/dev/gpiomem", O_RDWR | O_SYNC | O_CLOEXEC) ) >= 0) |
|
|
|
|
|
RASPBERRY_PI_PERI_BASE = 0 ; |
|
|
|
|
|
|
|
|
|
|
|
// ... otherwise fall back to the original /dev/mem which requires root level access |
|
|
|
|
|
|
|
|
|
|
|
else |
|
|
|
|
|
{ |
|
|
|
|
|
|
|
|
|
|
|
// This check is here because people are too stupid to check for themselves or read |
|
|
|
|
|
// error messages. |
|
|
|
|
|
|
|
|
|
|
|
if (geteuid () != 0) |
|
|
|
|
|
(void)wiringPiFailure (WPI_FATAL, "wiringPiSetup: Must be root. (Did you forget sudo?)\n") ; |
|
|
|
|
|
|
|
|
|
|
|
if ((fd = open ("/dev/mem", O_RDWR | O_SYNC | O_CLOEXEC) ) < 0) |
|
|
|
|
|
return wiringPiFailure (WPI_ALMOST, "wiringPiSetup: Unable to open /dev/mem: %s\n", strerror (errno)) ; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// Set the offsets into the memory interface. |
|
|
|
|
|
|
|
|
if ((fd = open ("/dev/mem", O_RDWR | O_SYNC | O_CLOEXEC) ) < 0) |
|
|
|
|
|
return wiringPiFailure (WPI_ALMOST, "wiringPiSetup: Unable to open /dev/mem: %s\n", strerror (errno)) ; |
|
|
|
|
|
|
|
|
GPIO_PADS = RASPBERRY_PI_PERI_BASE + 0x00100000 ; |
|
|
|
|
|
GPIO_CLOCK_BASE = RASPBERRY_PI_PERI_BASE + 0x00101000 ; |
|
|
|
|
|
GPIO_BASE = RASPBERRY_PI_PERI_BASE + 0x00200000 ; |
|
|
|
|
|
GPIO_TIMER = RASPBERRY_PI_PERI_BASE + 0x0000B000 ; |
|
|
|
|
|
GPIO_PWM = RASPBERRY_PI_PERI_BASE + 0x0020C000 ; |
|
|
|
|
|
|
|
|
// GPIO: |
|
|
|
|
|
|
|
|
// Map the individual hardware components |
|
|
|
|
|
|
|
|
|
|
|
// GPIO: |
|
|
|
|
|
|
|
|
gpio = (uint32_t *)mmap(0, BLOCK_SIZE, PROT_READ|PROT_WRITE, MAP_SHARED, fd, GPIO_BASE) ; |
|
|
gpio = (uint32_t *)mmap(0, BLOCK_SIZE, PROT_READ|PROT_WRITE, MAP_SHARED, fd, GPIO_BASE) ; |
|
|
if ((int32_t)gpio == -1) |
|
|
if ((int32_t)gpio == -1) |
|
|
return wiringPiFailure (WPI_ALMOST, "wiringPiSetup: mmap (GPIO) failed: %s\n", strerror (errno)) ; |
|
|
return wiringPiFailure (WPI_ALMOST, "wiringPiSetup: mmap (GPIO) failed: %s\n", strerror (errno)) ; |
|
|
|
|
|
|
|
|
// PWM |
|
|
|
|
|
|
|
|
// PWM |
|
|
|
|
|
|
|
|
pwm = (uint32_t *)mmap(0, BLOCK_SIZE, PROT_READ|PROT_WRITE, MAP_SHARED, fd, GPIO_PWM) ; |
|
|
pwm = (uint32_t *)mmap(0, BLOCK_SIZE, PROT_READ|PROT_WRITE, MAP_SHARED, fd, GPIO_PWM) ; |
|
|
if ((int32_t)pwm == -1) |
|
|
if ((int32_t)pwm == -1) |
|
|
return wiringPiFailure (WPI_ALMOST, "wiringPiSetup: mmap (PWM) failed: %s\n", strerror (errno)) ; |
|
|
return wiringPiFailure (WPI_ALMOST, "wiringPiSetup: mmap (PWM) failed: %s\n", strerror (errno)) ; |
|
|
|
|
|
|
|
|
// Clock control (needed for PWM) |
|
|
|
|
|
|
|
|
// Clock control (needed for PWM) |
|
|
|
|
|
|
|
|
clk = (uint32_t *)mmap(0, BLOCK_SIZE, PROT_READ|PROT_WRITE, MAP_SHARED, fd, CLOCK_BASE) ; |
|
|
|
|
|
|
|
|
clk = (uint32_t *)mmap(0, BLOCK_SIZE, PROT_READ|PROT_WRITE, MAP_SHARED, fd, GPIO_CLOCK_BASE) ; |
|
|
if ((int32_t)clk == -1) |
|
|
if ((int32_t)clk == -1) |
|
|
return wiringPiFailure (WPI_ALMOST, "wiringPiSetup: mmap (CLOCK) failed: %s\n", strerror (errno)) ; |
|
|
return wiringPiFailure (WPI_ALMOST, "wiringPiSetup: mmap (CLOCK) failed: %s\n", strerror (errno)) ; |
|
|
|
|
|
|
|
|
// The drive pads |
|
|
|
|
|
|
|
|
// The drive pads |
|
|
|
|
|
|
|
|
pads = (uint32_t *)mmap(0, BLOCK_SIZE, PROT_READ|PROT_WRITE, MAP_SHARED, fd, GPIO_PADS) ; |
|
|
pads = (uint32_t *)mmap(0, BLOCK_SIZE, PROT_READ|PROT_WRITE, MAP_SHARED, fd, GPIO_PADS) ; |
|
|
if ((int32_t)pads == -1) |
|
|
if ((int32_t)pads == -1) |
|
|
return wiringPiFailure (WPI_ALMOST, "wiringPiSetup: mmap (PADS) failed: %s\n", strerror (errno)) ; |
|
|
return wiringPiFailure (WPI_ALMOST, "wiringPiSetup: mmap (PADS) failed: %s\n", strerror (errno)) ; |
|
|
|
|
|
|
|
|
#ifdef USE_TIMER |
|
|
#ifdef USE_TIMER |
|
|
// The system timer |
|
|
|
|
|
|
|
|
// The system timer |
|
|
|
|
|
|
|
|
timer = (uint32_t *)mmap(0, BLOCK_SIZE, PROT_READ|PROT_WRITE, MAP_SHARED, fd, GPIO_TIMER) ; |
|
|
timer = (uint32_t *)mmap(0, BLOCK_SIZE, PROT_READ|PROT_WRITE, MAP_SHARED, fd, GPIO_TIMER) ; |
|
|
if ((int32_t)timer == -1) |
|
|
if ((int32_t)timer == -1) |
|
|