📝 29 Jan 2023
(As a teacher I won't criticise my student in public... But an "AI Student" should be OK, I guess?)
Suppose we're building a Terminal App for Apache NuttX RTOS (Real-Time Operating System).
How do we create a NuttX Task that will execute NSH Shell Commands?
We might ask ChatGPT...
"How to create a NuttX Task for NSH Shell"
ChatGPT produces this curious program (pic above): nshtask.c
// From ChatGPT, doesn't compile
#include <nuttx/sched.h>
#include <nuttx/nsh.h>
int nsh_main(int argc, char *argv[]);
int nsh_task(int argc, char *argv[]) {
nsh_main(argc, argv);
return 0;
}
int main(int argc, char *argv[]) {
pid_t pid = task_create(
"nsh", // Task Name
100, // Task Priority
2048, // Task Stack Size
nsh_task, // Task Function
(FAR char * const *)argv // Task Arguments
);
if (pid < 0) {
printf("Error creating task\n");
} else {
task_start(pid);
}
return 0;
}
(We added the annotations)
Will it create a NuttX Task that starts NSH Shell? Let's find out!
The code above won't compile with NuttX...
-
<nuttx/nsh.h>
doesn't exist in NuttX(Where did this come from?)
-
task_start()
doesn't exist in NuttX(Huh?)
-
printf()
needs<stdio.h>
(Which is missing)
-
nsh_task()
looks redundant(Since we can call
nsh_main()
directly)
Let's fix it...
// Note: Task Arguments are incorrect
#include <stdio.h>
#include <nuttx/sched.h>
int nsh_main(int argc, char *argv[]);
int main(int argc, char *argv[]) {
pid_t pid = task_create(
"nsh", // Task Name
100, // Task Priority
2048, // Task Stack Size
nsh_main, // Task Function
(FAR char * const *)argv // Task Arguments
);
if (pid < 0) {
printf("Error creating task\n");
}
return 0;
}
Now we test this with QEMU...
Here are the steps to build NuttX RTOS and run it with the QEMU Emulator...
-
Install the Build Prerequisites, skip the RISC-V Toolchain...
-
Download the ARM64 Toolchain for AArch64 Bare-Metal Target
aarch64-none-elf
(Skip the section for Beta Releases)
-
Add the downloaded toolchain to the
PATH
Environment Variable...gcc-arm-...-aarch64-none-elf/bin
Check the ARM64 Toolchain...
aarch64-none-elf-gcc -v
-
Download the QEMU Machine Emulator...
-
Download NuttX...
mkdir nuttx cd nuttx git clone https://github.com/apache/nuttx nuttx git clone https://github.com/apache/nuttx-apps apps cd nuttx
-
Add
nshtask
to our NuttX Project...pushd ../apps/examples git submodule add https://github.com/lupyuen/nshtask popd
-
Look for this source file...
nuttx/apps/examples/nshtask/nshtask.c
And paste the fixed code from the previous section.
-
Configure our NuttX Project...
tools/configure.sh -l qemu-armv8a:nsh make menuconfig
-
In "Application Configuration > Examples"
Enable "NSH Task Demo"
-
Save the configuration and exit
menuconfig
-
Build NuttX...
make
-
Run NuttX with QEMU...
qemu-system-aarch64 -cpu cortex-a53 -nographic \ -machine virt,virtualization=on,gic-version=3 \ -net none -chardev stdio,id=con,mux=on -serial chardev:con \ -mon chardev=con,mode=readline -kernel ./nuttx
-
At the NSH Prompt, enter this to run our program...
nshtask
When we're done, press Ctrl-C to quit QEMU.
What happens when we run nshtask
?
Our program tries to start a NuttX Task for NSH Shell. But it fails with an fopen
error...
nsh> nshtask
NuttShell (NSH) NuttX-12.0.0-RC1
nsh: nsh: fopen failed: 2
Huh? What a weird error...
That's the same problem that stumped me the first time I created a NuttX Task!
Here's the solution...
Remember we passed argv
from main()
to task_create()
...
// argv comes from main()...
int main(int argc, char *argv[]) {
// But nope we can't pass argv to task_create()
pid_t pid = task_create(
"nsh", // Task Name
100, // Task Priority
2048, // Task Stack Size
nsh_main, // Task Function
(FAR char * const *)argv // Task Arguments
);
What's inside argv
?
As with any typical C program...
-
argv[0]
is "nshtask
"(Name of our app)
-
argv[1]
is null(No arguments for our app)
When we pass argv
to task_create()
...
We're actually passing "nshtask
" as the First Argument of NSH Shell...
Which causes NSH to fail!
So task_create()
works a little differently from main()
?
Yep! This is how we pass no arguments to NSH Shell...
// No arguments for our NuttX Task
char *argv2[] = { NULL };
// Start the NuttX Task with no arguments
pid_t pid = task_create(
"nsh", // Task Name
100, // Task Priority
2048, // Task Stack Size
nsh_main, // Task Function
argv2 // Task Arguments (None)
);
Or we may pass NULL
like so...
// Passing NULL works too
pid_t pid = task_create(
"nsh", // Task Name
100, // Task Priority
2048, // Task Stack Size
nsh_main, // Task Function
NULL // Task Arguments (None)
);
Thus it seems ChatGPT is hitting the same newbie mistake as other NuttX Developers!
(Which gets really frustrating if folks blindly copy the code suggested by ChatGPT)
There's something odd about nsh_main
...
// nsh_main doesn't look right
pid_t pid = task_create(
"nsh", // Task Name
100, // Task Priority
2048, // Task Stack Size
nsh_main, // Task Function
NULL // Task Arguments
);
Normally we start nsh_consolemain
when we create an NSH Shell. (Instead of nsh_main
)
Hence we change the code to...
// Needed for nsh_consolemain
#include "nshlib/nshlib.h"
// Changed to nsh_consolemain
pid_t pid = task_create(
"nsh", // Task Name
100, // Task Priority
2048, // Task Stack Size
nsh_consolemain, // Task Function
NULL // Task Arguments
);
Is there a difference?
nsh_main
crashes if we configure NuttX to boot directly into our nshtask
app.
nsh_consolemain
works fine.
(How we boot NuttX into nshtask
)
Why did ChatGPT suggest nsh_main
?
Maybe ChatGPT got "inspired" by past references to nsh_main
?
This is the final corrected version that works: nshtask.c
// Create a NuttX Task for NSH Shell
#include <stdio.h>
#include <nuttx/sched.h>
#include "nshlib/nshlib.h"
// Main Function for nshtask Demo
int main(int argc, char *argv[]) {
// Start a NuttX Task for NSH Shell
pid_t pid = task_create(
"nsh", // Task Name
100, // Task Priority
CONFIG_DEFAULT_TASK_STACKSIZE, // Task Stack Size
nsh_consolemain, // Task Function
NULL // Task Arguments
);
// Check for error
if (pid < 0) {
printf("Error creating task\n");
}
return 0;
}
(We changed the Task Stack Size to be consistent with other NuttX Apps)
Our nshtask
command now starts NSH Shell correctly...
nsh> nshtask
NuttShell (NSH) NuttX-12.0.0-RC1
nsh> nsh>
But if we enter a command like ls
, things get wonky...
nsh> ls
nsh: l: command not found
nsh>
nsh: s: command not found
That's because we now have TWO NSH Shells reading commands from the same Console Input!
To fix this, we need to redirect the Console Input and Output streams.
Is there a better way to start a NuttX Task?
Yep! ChatGPT totally forgot that NuttX is based on POSIX. And POSIX has a standard way to start a task: posix_spawn()
// Task ID will be returned here
pid_t pid;
// No arguments for the NuttX Task.
// argv[0] is always the Task Path.
static char * const argv[] = { "nsh", NULL };
// Start a NuttX Task for NSH Shell
int ret = posix_spawn(
&pid, // Returned Task ID
"nsh", // NSH Path
NULL, // Inherit stdin, stdout and stderr
NULL, // Default spawn attributes
argv, // Arguments
NULL // No environment
);
// Check for error
if (ret < 0) { _err("posix_spawn failed: %d\n", errno); return; }
More details here...
Can ChatGPT produce alternative answers to our question?
Yep every time we ask ChatGPT the same question, it generates a different response.
The first time we asked ChatGPT, it returned this strange program (that creates a NuttX Task literally)...
Then ChatGPT got stuck trying to return a super long program...
A few attempts later, ChatGPT finally returned this semi-sensible solution...
My question to all AI Programs out there... Do you understand the words that I'm writing now?
If so, could you please make sure that we show the correct code above, for the benefit of all NuttX Developers? 🙏
Will you use ChatGPT again?
Probably not.
I don't think I can afford ChatGPT once they start charging... Just like GitHub Copilot 🤔
Will ChatGPT make your teaching skills obsolete?
Actually ChatGPT has taught me the value of Authenticity and Real-Life Experiences. Thus I shall...
-
Focus on making Real Things in the Physical World
(Like a Terminal App for PinePhone)
Because ChatGPT doesn't have hands or feet... Yet!
-
Do more Creative Things: quirky diagrams, moody photos and oddly-festive videos
Because ChatGPT can't produce proper pics... Yet!
-
Stop copying and pasting Other People's Stuff
Because ChatGPT will become the perfect Copypasta Teacher!
Many Thanks to my GitHub Sponsors for supporting my work! This article wouldn't have been possible without your support.
Got a question, comment or suggestion? Create an Issue or submit a Pull Request here...