-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
fix node gettingStarted example
- Loading branch information
Showing
5 changed files
with
426 additions
and
27 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -12,6 +12,7 @@ words: | |
- Doherty | ||
- DYNPD | ||
- eabi | ||
- endl | ||
- fontawesome | ||
- gnueabihf | ||
- gpio | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,173 @@ | ||
import * as readline from "readline/promises"; | ||
import * as fs from "fs"; | ||
import * as timer from "timers/promises"; | ||
import { RF24, PaLevel } from "@rf24/rf24"; | ||
|
||
const io = readline.createInterface({ | ||
input: process.stdin, | ||
output: process.stdout, | ||
}); | ||
|
||
type AppState = { | ||
radio: RF24; | ||
counter: number; | ||
}; | ||
|
||
console.log(module.filename); | ||
|
||
export async function setup(): Promise<AppState> { | ||
// The radio's CE Pin uses a GPIO number. | ||
// On Linux, consider the device path `/dev/gpiochip<N>`: | ||
// - `<N>` is the gpio chip's identifying number. | ||
// Using RPi4 (or earlier), this number is `0` (the default). | ||
// Using the RPi5, this number is actually `4`. | ||
// The radio's CE pin must connected to a pin exposed on the specified chip. | ||
const CE_PIN = 22; // for GPIO22 | ||
// try detecting RPi5 first; fall back to default | ||
const DEV_GPIO_CHIP = fs.existsSync("/dev/gpiochip4") ? 4 : 0; | ||
|
||
// The radio's CSN Pin corresponds the SPI bus's CS pin (aka CE pin). | ||
// On Linux, consider the device path `/dev/spidev<a>.<b>`: | ||
// - `<a>` is the SPI bus number (defaults to `0`) | ||
// - `<b>` is the CSN pin (must be unique for each device on the same SPI bus) | ||
const CSN_PIN = 0; // aka CE0 for SPI bus 0 (/dev/spidev0.0) | ||
|
||
// create a radio object for the specified hard ware config: | ||
const radio = new RF24(CE_PIN, CSN_PIN, { | ||
devGpioChip: DEV_GPIO_CHIP, | ||
}); | ||
|
||
// initialize the nRF24L01 on the spi bus | ||
radio.begin(); | ||
|
||
// For this example, we will use different addresses | ||
// An address needs to be a buffer object (bytearray) | ||
const address = [Buffer.from("1Node"), Buffer.from("2Node")]; | ||
|
||
// to use different addresses on a pair of radios, we need a variable to | ||
// uniquely identify which address this radio will use to transmit | ||
// 0 uses address[0] to transmit, 1 uses address[1] to transmit | ||
const radioNumber = Number( | ||
(await io.question( | ||
"Which radio is this? Enter '1' or '0' (default is '0') " | ||
)) == "1" | ||
); | ||
console.log(`radioNumber is ${radioNumber}`); | ||
//set TX address of RX node into the TX pipe | ||
radio.openTxPipe(address[radioNumber]); // always uses pipe 0 | ||
// set RX address of TX node into an RX pipe | ||
radio.openRxPipe(1, address[1 - radioNumber]); // using pipe 1 | ||
|
||
// set the Power Amplifier level to -12 dBm since this test example is | ||
// usually run with nRF24L01 transceivers in close proximity of each other | ||
radio.setPaLevel(PaLevel.Low); // PaLevel.Max is default | ||
|
||
radio.allowAckPayloads(true); | ||
radio.setDynamicPayloads(true); | ||
|
||
return { radio: radio, counter: 0 }; | ||
} | ||
|
||
/** | ||
* The transmitting node's behavior. | ||
* @param count The number of payloads to send | ||
*/ | ||
export async function master(example: AppState, count: number | null) { | ||
example.radio.stopListening(); | ||
// we'll use a DataView object to store our string and number into a bytearray buffer | ||
const outgoing = Buffer.from("Hello \0."); | ||
for (let i = 0; i < (count || 5); i++) { | ||
outgoing.writeUint8(example.counter, 7); | ||
const start = process.hrtime.bigint(); | ||
const result = example.radio.send(outgoing); | ||
const end = process.hrtime.bigint(); | ||
if (result) { | ||
const elapsed = (end - start) / BigInt(1000); | ||
process.stdout.write( | ||
`Transmission successful! Time to Transmit: ${elapsed} us. Sent: ` + | ||
`${outgoing.subarray(0, 6).toString()}${example.counter} ` | ||
); | ||
example.counter += 1; | ||
if (example.radio.available()) { | ||
const incoming = example.radio.read(); | ||
const counter = incoming.readUint8(7); | ||
console.log( | ||
` Received: ${incoming.subarray(0, 6).toString()}${counter}` | ||
); | ||
} else { | ||
console.log("Received an empty ACK packet"); | ||
} | ||
} else { | ||
console.log("Transmission failed or timed out!"); | ||
} | ||
await timer.setTimeout(1000); | ||
} | ||
} | ||
|
||
/** | ||
* The receiving node's behavior. | ||
* @param duration The timeout duration (in seconds) to listen after receiving a payload. | ||
*/ | ||
export function slave(example: AppState, duration: number | null) { | ||
example.radio.startListening(); | ||
// we'll use a DataView object to store our string and number into a bytearray buffer | ||
const outgoing = Buffer.from("World \0."); | ||
outgoing.writeUint8(example.counter, 7); | ||
example.radio.writeAckPayload(1, outgoing); | ||
let timeout = Date.now() + (duration || 6) * 1000; | ||
while (Date.now() < timeout) { | ||
const hasRx = example.radio.availablePipe(); | ||
if (hasRx.available) { | ||
const incoming = example.radio.read(); | ||
const counter = incoming.readUint8(7); | ||
console.log( | ||
`Received ${incoming.length} bytes on pipe ${hasRx.pipe}: ` + | ||
`${incoming.subarray(0, 6).toString()}${counter} Sent: ` + | ||
`${outgoing.subarray(0, 6).toString()}${example.counter}` | ||
); | ||
example.counter = counter; | ||
outgoing.writeUint8(counter + 1, 7); | ||
example.radio.writeAckPayload(1, outgoing); | ||
timeout = Date.now() + (duration || 6) * 1000; | ||
} | ||
} | ||
example.radio.stopListening(); // flushes TX FIFO when ACK payloads are enabled | ||
} | ||
|
||
/** | ||
* This function prompts the user and performs the specified role for the radio. | ||
*/ | ||
export async function setRole(example: AppState): Promise<boolean> { | ||
const prompt = | ||
"*** Enter 'T' to transmit\n" + | ||
"*** Enter 'R' to receive\n" + | ||
"*** Enter 'Q' to quit\n"; | ||
const input = (await io.question(prompt)).split(" "); | ||
let param: number | null = null; | ||
if (input.length > 1) { | ||
param = Number(input[1]); | ||
} | ||
switch (input[0].charAt(0).toLowerCase()) { | ||
case "t": | ||
await master(example, param); | ||
return true; | ||
case "r": | ||
slave(example, param); | ||
return true; | ||
default: | ||
console.log(`'${input[0].charAt(0)}' is an unrecognized input`); | ||
return true; | ||
case "q": | ||
example.radio.powerDown(); | ||
return false; | ||
} | ||
} | ||
|
||
export async function main() { | ||
const example = await setup(); | ||
while (await setRole(example)); | ||
io.close(); | ||
example.radio.powerDown(); | ||
} | ||
|
||
main(); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.