Working with Buildroot to create, compile and run custom programs on target hardware...
Naveen Kumar Gutti
Cutting edge innovations and solutions provider - Qualcomm WIFI | Mediatek Platform | Openwrt | Linux Application & Kernel Programming | Netfilter | L3,L4 protocols | AI&ML | LXC | Data Security | Web3
In the previous article, we explored the process of building a Cross-Compilation toolchain using Buildroot. We delved into configuring the toolchain by selecting crucial parameters such as the CPU Architecture (for example, Cortex A53 for our target hardware), ensuring the correct Endianness for the target platform, opting for the MUSL C Library, and choosing GCC 12.x Version as our compiler, complete with c++ support.
Moving Forward: Compiling Custom Programs
With the generated toolchain in hand, it's now time to delve into the next phase of our journey. We'll explore compiling our custom programs, providing us with a hands-on experience of the entire process. This step is integral to gaining a deeper understanding of the toolchain's capabilities and honing our skills in the cross-compilation domain.
Our journey continues as we dive into practical use cases, applying the knowledge gained so far through coding examples. This hands-on approach will enhance our understanding and proficiency in cross-compilation. Let's take a look at several use cases, each accompanied by detailed coding examples.
Estimated time to complete this article with hands-on experience: 2 hours
Creating a custom package in Buildroot:
In the next phase of our exploration, we will delve into the creation of a custom package housing the code essential for developing the use cases outlined in this article. This package will serve as a structured and reusable foundation for our ongoing development endeavours.
Step 1: Create package structure
Explore the sample package structure illustrated in the image below. Tailor the parameters to align with the specific requirements of your development.
Directory Structure:
# create the directory structure of the custom package
cd <buildroot_dir>/package
mkdir sammytechplayground
touch sammytechplayground/Config.in
touch sammytechplayground/sammytechplayground.mk
mkdir sammytechplayground/src
touch sammytechplayground/src/Makefile
touch sammytechplayground/src/prog1.c
touch sammytechplayground/src/read_data.json
Following are the content of the files mentioned above in the directory structure
Config.in:
config BR2_PACKAGE_SAMMYTECHPLAYGROUND
bool "Sammy Tech Playground"
help
Mypackage - This is a package containing the coding samples
for Sammy Tech Playground buildroot system
sammytechplayground.mk:
SAMMYTECHPLAYGROUND_VERSION=1.0
SAMMYTECHPLAYGROUND_SITE=package/sammytechplayground/src
SAMMYTECHPLAYGROUND_SITE_METHOD=local
define SAMMYTECHPLAYGROUND_BUILD_CMDS
$(MAKE) CC="$(TARGET_CC)" LD="$(TARGET_LD)" AR="$(TARGET_AR)" -C $(@D)
endef
define SAMMYTECHPLAYGROUND_INSTALL_TARGET_CMDS
endef
$(eval $(generic-package))
src/Makefile:
all:
$(CC) prog1.c -o prog1
clean:
rm -rf prog1
prog1.c:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(int argc, char **argv){
printf("Hello everyone, wish you all the best and success always - sammytechplayground\n");
return 0;
}
<buildroot_dir>/package/Config.in:
Add the following lines to the end of the file before "endmenu" keyword. Refer to the attached screenshot...
menu "SammyTechPlayground"
source "package/sammytechplayground/Config.in"
endmenu
Compiling and Testing the package:
Once you have completed all the above steps, your package will be recognised by the Buildroot environment as shown in the image below
cd <buildroot_dir>
make menuconfig
Target Packages --->
SammyTechPlayground --->
[*] Sammy Tech Playground
To compile and generate the binaries, use the following commands
# inside <buildroot_dir>/
make sammytechplayground V=s
# if you want to recompile the package and regenerate the binaries, # then use these commands
make sammytechplayground-dirclean
make sammytechplayground V=s
Once the compilation is success, target binaries which are cross compiled will be generated at the following location
ls -l output/build/sammytechplayground-1.0/prog1
Confirm the target binaries:
To confirm that the toolchain has indeed performed cross-compilation and generated for target hardware, lets verify the generated binary with the following command
file output/build/sammytechplayground-1.0/prog1
The output should be as follows indicating that cross compiled binary is generated for aarch64 architecture
output/build/sammytechplayground-1.0/prog1: ELF 64-bit LSB pie executable, ARM aarch64, version 1 (SYSV), dynamically linked, interpreter /lib/ld-musl-aarch64.so.1, not stripped
Testing the binaries:
For this article purpose, I am using Banana PI R3 as the target hardware. After the binary is generated, transfer the binary to target hardware as follows
scp output/build/sammytechplayground-1.0/prog1 [email protected]:~
After transfer is successfull, login to the target hardware and execute the program as shown below
This concludes the package creation, compilation, binary generation and testing on target hardware.
Congratulations on Achieving the Initial Step in Cross-Compilation
Kudos ?? on reaching the first milestone in the cross-compilation journey for your custom program, geared to run on specialized hardware. This accomplishment marks a significant step forward, and your dedication to the cross-compilation process sets the stage for further development and exploration. Well done!!!
Exploration of Diverse Use Cases: Integrating Dependencies into the Custom Package
Our journey takes a deeper plunge as we delve into various use cases, unraveling the intricacies of incorporating dependencies into our custom package. This phase is pivotal as we enhance our understanding and proficiency while bringing our custom use cases to fruition. Let's navigate through these scenarios and seamlessly integrate dependencies into our custom package to accomplish our objectives.
Usecase 1: Exploring JSON Interaction in C: Reading and Writing Files
In today's world, JSON (JavaScript Object Notation) data stands out as the most widely used format for exchanging data between various entities. As software developers, we encounter JSON in our daily lives. But have you ever wondered about the origin of JSON?
Let's take a moment to appreciate that JSON Format was invented by none other than "Douglas Crockford." He is an American computer programmer with notable involvement in the development of the JavaScript language. As we delve into the intricacies of JSON, it's enlightening to know that its roots trace back to the contributions of Douglas Crockford, a key figure in the realm of computer programming.
More information at: https://en.wikipedia.org/wiki/Douglas_Crockford#:~:text=Douglas%20Crockford%20is%20an%20American,analyzer%20JSLint%20and%20minifier%20JSMin .
In this section, our focus shifts to the practical application of JSON manipulation through the execution of two distinct use cases:
Dependencies in Buildroot to support JSON Interaction:
Before we begin interacting with JSON, we need to add dependency of the JSON package to our custom package. For our purpose we are using "json-c" library.
This can be done by adding the following lines into our Config.in file and sammytechplayground.mk files, as shown in the image below
领英推荐
Use case 1.1 - JSON Reading:
Create another new program "prog2_json_read.c" as given below in the src directory.
prog2_json_read.c
/*
* Program to parse the JSON file
*
*/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <json-c/json.h>
int main(int argc, char **argv){
FILE *fp = NULL;
char buffer[1024] = {0};
struct json_object *main_obj = NULL;
struct json_object *data_obj = NULL;
printf("Hello Sammy'sTechPlayground\n");
// read the entire file data
fp = fopen("read_data.json", "r");
if(fp){
fread(buffer, sizeof(buffer), 1024, fp);
fclose(fp);
}
printf("File data: %s\n", buffer);
// parse the json object
main_obj = json_tokener_parse(buffer);
if(main_obj){
printf("Parsing success\n");
if(json_object_object_get_ex(main_obj, "name", &data_obj)){
printf("Name from JSON: %s\n", json_object_get_string(data_obj));
}
if(json_object_object_get_ex(main_obj, "passion", &data_obj)){
printf("Passion from JSON: %s\n", json_object_get_string(data_obj));
}
if(json_object_object_get_ex(main_obj, "linkedin-url", &data_obj)){
printf("LinkedIn URL from JSON: %s\n", json_object_get_string(data_obj));
}
json_object_put(main_obj);
}
return 0;
}
In order for this program to read and display the contents of JSON file, we need to create a sample data file as given below. Save it as "read_data.json" in the src directory
{"name":"naveen", "passion":"Software development and developing cutting edge solutions", "linkedin-url":"https://www.dhirubhai.net/in/naveen-kumar-gutti"}
Modify the Makefile as follows
all:
$(CC) prog1.c -o prog1
$(CC) prog2_json_read.c -o prog2_json_read -ljson-c
clean:
rm -rf prog1 prog2_json_read
After these changes, recompile the package as follows
make sammytechplayground-dirclean
make sammytechplayground V=s
After compilation is success, transfer the prog2_json_read binary and read_data.json to the target hardware as shown in the image below
Post-Compilation Execution Result: A Glimpse Through the Screenshot
Embark on the testing phase and relish the satisfaction of a small triumph. This step is crucial in validating the functionality and performance of your program. Enjoy the process of testing, knowing that each successful test brings you closer to your ultimate goal.
Usecase 1.2 - JSON Generation
Create another program "prog3_json_create.c" in the src directory and modify Makefile to include compilation of prog3_json_create.c
prog3_json_create.c
/*
* Program to create the JSON file
*
*/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <json-c/json.h>
// structure to host sample data
struct sample_data {
int id; //identifier of the data
char name[128]; // name of the person
char designation[128]; // designation of the person
char address[256]; // address of the person
};
int main(int argc, char **argv){
int i = 0;
struct sample_data data_obj[3] = {0};
struct json_object *main_obj = NULL;
for (i=0; i < 3; i++){
memset(&data_obj[i], 0, sizeof(struct sample_data));
data_obj[i].id = i + 1;
snprintf(data_obj[i].name, sizeof(data_obj[i].name), "DemoUser%d", i + 1);
snprintf(data_obj[i].designation, sizeof(data_obj[i].designation), "DemoDesignation%d", i + 1);
snprintf(data_obj[i].address, sizeof(data_obj[i].address), "Address%d", i + 1);
}
// create the json object from scratch
main_obj = json_object_new_object();
if(main_obj){
// create a json array object
struct json_object *arr_obj = json_object_new_array();
if(arr_obj){
// create the individual json objects for struct sample_data array and add to json array
for(i=0; i < 3; i++){
struct json_object *user_obj = json_object_new_object();
json_object_object_add(user_obj, "id", json_object_new_int(data_obj[i].id));
json_object_object_add(user_obj, "name", json_object_new_string(data_obj[i].name));
json_object_object_add(user_obj, "designation", json_object_new_string(data_obj[i].designation));
json_object_object_add(user_obj, "address", json_object_new_string(data_obj[i].address));
json_object_array_add(arr_obj, user_obj);
}
json_object_object_add(main_obj, "data", arr_obj);
}
// print the generated JSON string
printf("Result: %s\n", json_object_to_json_string(main_obj));
// Write the data to output file
FILE *fp = NULL;
fp = fopen("sample_data.json", "w");
if(fp){
fprintf(fp, "%s", json_object_to_json_string(main_obj));
fclose(fp);
}
json_object_put(main_obj);
}
return 0;
}
Post-Compilation Execution Result: A Glimpse Through the Screenshot
Upon successful compilation and transfer of the program to the target hardware, the following screenshot captures the outcome of its execution. Take a moment to examine the displayed results, a tangible representation of the program in action on the designated hardware. This visual insight provides valuable feedback and serves as a checkpoint in your cross-compilation journey.
Conclusion of the First Use Case: Cross-Compilation Sample Program for JSON Interaction in C
With this, we wrap up the initial use case, successfully cross-compiling a sample program designed to interact with JSON objects using the C programming language.
Usecase 2: Exploring YAML Interaction in C: Reading and Writing Files
YAML (YAML Ain't Markup Language) was invented by Clark Evans and Ingy d?t Net.
They developed YAML with the goal of creating a human-readable data serialization format that is easy to write and understand. YAML is often used for configuration files, data exchange between languages with different data structures, and other scenarios where human readability is crucial. The official specification for YAML is maintained by the YAML community.
In this use case, we delve into the methodology of interacting with YAML files using the C language. Our focus will encompass both reading from a YAML file and creating/writing to a YAML file. Join us in uncovering the intricacies of YAML file manipulation in the C programming language for a comprehensive understanding of these essential operations.
Enabling libyaml Library for YAML Support in Our Package
To facilitate YAML-based interaction in our custom package, it's imperative to incorporate the "libyaml" library into the Buildroot environment. Following the compilation, the next step involves transferring the libyaml libraries to the target hardware.
Steps:
After compilation, transfer the necessary libraries to the target hardware as shown in the image below
Usecase 2.1: Reading a YAML file
Create a source file with the name of prog4_yaml_read.c and add the same to the Makefile
Following is the code to parse YAML file and print the data
prog4_yaml_read.c
// Program to read and parse the YAML file based on tokens
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <yaml.h>
struct address {
char city[128];
char state[128];
char country[128];
};
struct person {
char name[128];
int age;
struct address addr;
};
int main(int argc, char **argv){
FILE *fp = NULL;
struct person obj;
memset(&obj, 0, sizeof(struct person));
fp = fopen("read_data.yaml", "r");
if(!fp){
printf("Failed to open read_data.yaml file\n");
return -1;
}
yaml_parser_t yaml_parser;
yaml_token_t token;
// Initialize the yaml parser object
if(!yaml_parser_initialize(&yaml_parser)){
printf("Failed to create parser object\n");
return -1;
}
// set the file as parse source
yaml_parser_set_input_file(&yaml_parser, fp);
char data_type[128]={0};
char placeholder[128] = {0};
do {
yaml_parser_scan(&yaml_parser, &token);
switch(token.type)
{
case YAML_STREAM_START_TOKEN:
break;
case YAML_STREAM_END_TOKEN:
break;
case YAML_KEY_TOKEN:
memset(data_type, 0, sizeof(data_type));
strcpy(data_type, "key");
break;
case YAML_VALUE_TOKEN:
memset(data_type, 0, sizeof(data_type));
strcpy(data_type, "value");
break;
case YAML_BLOCK_SEQUENCE_START_TOKEN:
break;
case YAML_BLOCK_ENTRY_TOKEN:
break;
case YAML_BLOCK_END_TOKEN:
break;
case YAML_BLOCK_MAPPING_START_TOKEN:
break;
case YAML_SCALAR_TOKEN:
if(!strncmp(data_type, "key", 3)){
// current data is key, so copy to placeholder
memset(placeholder, 0, sizeof(placeholder));
strcpy(placeholder, token.data.scalar.value);
}
if(!strncmp(data_type, "value", 5)){
if(!strncmp(placeholder, "name", 4)){
strcpy(obj.name, token.data.scalar.value);
}
if(!strncmp(placeholder, "age", 3)){
obj.age = atoi(token.data.scalar.value);
}
if(!strncmp(placeholder, "city", 4)){
strcpy(obj.addr.city, token.data.scalar.value);
}
if(!strncmp(placeholder, "state", 5)){
strcpy(obj.addr.state, token.data.scalar.value);
}
if(!strncmp(placeholder, "country", 7)){
strcpy(obj.addr.country, token.data.scalar.value);
}
}
break;
default:
break;
}
if(token.type != YAML_STREAM_END_TOKEN)
yaml_token_delete(&token);
} while(token.type != YAML_STREAM_END_TOKEN);
// delete the token object
yaml_token_delete(&token);
// print the result
printf("yaml parsing completed, printing struct data\n");
printf("Name: %s, age: %d, city: %s, state: %s, country: %s\n",
obj.name,
obj.age,
obj.addr.city,
obj.addr.state,
obj.addr.country);
cleanup:
// Delete the yaml parser object
yaml_parser_delete(&yaml_parser);
if(fp)
fclose(fp);
return 0;
}
Post-Compilation Execution Result: A Glimpse Through the Screenshot
Upon successful compilation and transfer of the program to the target hardware, the following screenshot captures the outcome of its execution
Usecase 2.1: YAML generation
Create a source file with the name of "prog5_yaml_write.c" and add the same to the Makefile
Following is the code to generate YAML data and write to a file
prog5_yaml_write.c
// Program to write the YAML file from C
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <yaml.h>
int main(int argc, char **argv){
FILE *fp = NULL;
fp = fopen("sample_data.yaml", "w");
if(!fp){
printf("Failed to open the target file\n");
return -1;
}
yaml_emitter_t emitter;
yaml_emitter_initialize(&emitter);
yaml_emitter_set_output_file(&emitter, fp);
yaml_event_t event;
yaml_stream_start_event_initialize(&event, YAML_UTF8_ENCODING);
yaml_emitter_emit(&emitter, &event);
yaml_document_start_event_initialize(&event, NULL, NULL, NULL, 0);
yaml_emitter_emit(&emitter, &event);
yaml_mapping_start_event_initialize(&event, NULL, (yaml_char_t*)YAML_MAP_TAG, 1, YAML_ANY_MAPPING_STYLE);
yaml_emitter_emit(&emitter, &event);
yaml_scalar_event_initialize(&event, NULL, NULL, (yaml_char_t *)"name", -1, 1, 0, YAML_PLAIN_SCALAR_STYLE);
yaml_emitter_emit(&emitter, &event);
yaml_scalar_event_initialize(&event, NULL, NULL, (yaml_char_t *)"Naveen From C", -1, 1, 0, YAML_PLAIN_SCALAR_STYLE);
yaml_emitter_emit(&emitter, &event);
yaml_scalar_event_initialize(&event, NULL, NULL, (yaml_char_t *)"age", -1, 1, 0, YAML_PLAIN_SCALAR_STYLE);
yaml_emitter_emit(&emitter, &event);
yaml_scalar_event_initialize(&event, NULL, NULL, (yaml_char_t *)"38", -1, 1, 0, YAML_PLAIN_SCALAR_STYLE);
yaml_emitter_emit(&emitter, &event);
// End mapping node
yaml_mapping_end_event_initialize(&event);
yaml_emitter_emit(&emitter, &event);
yaml_document_end_event_initialize(&event, 0);
yaml_emitter_emit(&emitter, &event);
// End YAML document
yaml_stream_end_event_initialize(&event);
yaml_emitter_emit(&emitter, &event);
yaml_emitter_flush(&emitter);
// delete the emitter object
yaml_emitter_delete(&emitter);
printf("written\n");
if(fp)
fclose(fp);
return 0;
}
Post-Compilation Execution Result: A Glimpse Through the Screenshot
Upon successful compilation and transfer of the program to the target hardware, the following screenshot captures the outcome of its execution
Conclusion: Cross Compilation with Buildroot for Target Hardware
In wrapping up this article, we've explored the process of crafting custom programs through cross-compilation for target hardware within the Buildroot environment. Additionally, we delved into methods for interacting with JSON files using the json-c library and YAML files using the libyaml library.
I trust you found this article informative and engaging. If there are any aspects you'd like to see further addressed or expanded upon, please do let me know. Stay tuned for upcoming articles as we continue to explore intriguing topics.
Thank you to everyone, If you found this information valuable, consider subscribing to my profile for updates on future articles. Looking forward to bringing you more insightful content in the near future. See you soon and I wish you all the best on your endeavuors!
You can find all the source code at my github repository by visiting the link below
References: