Working with Buildroot to create, compile and run custom programs on target hardware...

Working with Buildroot to create, compile and run custom programs on target hardware...


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 of Custom Package

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
        
Enabling package - sammytechplayground in buildroot

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

Package help text in buildroot
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

Program running on BananaPI Hardware - Cross compiled with Buildroot

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:

  1. Parsing a JSON file
  2. Creating a JSON file

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

Config.in
sammytechplayground.mk


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

Recompiling and transferring the binary and json file onto target hardware

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.

Execution of cross compilation program on target hardware to read json data

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.

Result of JSON object generation and writing the JSON object to file

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.

https://en.wikipedia.org/wiki/YAML#:~:text=YAML%20(%2F%CB%88j%C3%A6m%C9%99l%2F%2C%20rhymes%20with,Net%20and%20Oren%20Ben%2DKiki.

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:

  1. Modify our package to include dependencies as follows which auto selects "libyaml" package.select BR2_PACKAGE_LIBYAML - Add this line into Config.in of sammytechplayground packageSAMMYTECHPLAYGROUND_DEPENDENCIES=json-c libyaml - Add this line into sammytechplayground.mk file in sammytechplayground package
  2. make menuconfig - to confirm the libyaml package is auto selected
  3. make libyaml V=s - To compile and build the libyaml package

After compilation, transfer the necessary libraries to the target hardware as shown in the image below

Transfer libyaml libraries to target hardware


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

YAML Reading from C Program using libyaml library



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

Generation of YAML file from C program using libyaml library

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

https://github.com/gvvsnrnaveen/buildroot.git


References:

  1. https://en.wikipedia.org/wiki/YAML#:~:text=YAML%20(%2F%CB%88j%C3%A6m%C9%99l%2F%2C%20rhymes%20with,Net%20and%20Oren%20Ben%2DKiki.
  2. https://en.wikipedia.org/wiki/Douglas_Crockford#:~:text=Douglas%20Crockford%20is%20an%20American,analyzer%20JSLint%20and%20minifier%20JSMin .
  3. https://buildroot.org/downloads/manual/manual.html#adding-packages
  4. https://buildroot.org/downloads/manual/manual.html#_the_literal_mk_literal_file
  5. https://buildroot.org/downloads/manual/manual.html#_config_files
  6. https://wiki.banana-pi.org/Banana_Pi_BPI-R3

要查看或添加评论,请登录

Naveen Kumar Gutti的更多文章

社区洞察

其他会员也浏览了