Skip to content

Latest commit

 

History

History
382 lines (294 loc) · 10.4 KB

README.md

File metadata and controls

382 lines (294 loc) · 10.4 KB

Developer guide

You can build your own firmware using the ESP-IDF framework or Arduino. However, we will focus on the ESP-IDF framework in this guide.

Table of contents

Prerequisites

Development setup

  1. Clone this repository:

Using HTTPS:

git clone https://github.com/ElectronicCats/badge-EKOParty-2024

Using SSH:

git clone [email protected]:ElectronicCats/badge-EKOParty-2024.git
  1. Change to the project directory:
cd badge-EKOParty-2024
  1. Get the submodules:
git submodule update --init --recursive
  1. Change to the firmware directory:
cd firmware
  1. Set the IDF_PATH environment variable:
get_idf

Note: get_idf should be an alias you created if you followed the ESP-IDF installation guide.

  1. Set the Minino environment variable. This project uses the Minino firmware codebase. You have to clone it somewhere in your computer and get the path to its firmware directory. For example:
/Users/user/Minino/firmware

Set the Minino environment variable:

export MININO_PATH=/Users/user/Minino/firmware

Note: Replace /Users/user/Minino/firmware with the path to your Minino firmware directory, not the root directory of the Minino repository.

  1. Go to the Minino firmware directory and change to the eko branch:
cd $MININO_PATH
git checkout eko
  1. Go back to this project directory:
cd -
  1. Setup the project environment:
make setup

Building the firmware

To build the firmware, run the following command:

make compile

Flashing the firmware

  1. Connect your Minino to your computer and identify the port it is connected to. For example, /dev/ttyUSB0 or /dev/ttyACM0 on Linux, COM1 or COM2 on Windows, or /dev/cu.usbmodem14401 on macOS.

  2. Set the PORT to an environment variable, for example:

export PORT=/dev/ttyUSB0

Replace /dev/ttyUSB0 with the port your Minino is connected to.

  1. Flash the firmware:
make flash

Monitoring the serial output

To monitor the serial output, run the following command:

make monitor

Full build process

To do all the previous steps in one command, run:

make all

Cleaning the project

If you you have dependencies issues or want to clean the project, run the following command:

make clean

Create a release

To create a release, you need to update the firmware version in the menuconfig. To do this, run:

idf.py menuconfig

Navigate to General project -> Firmware version and update the version number.

Get the build files:

./get_build.sh

If you can't run the script, make sure it has the correct permissions:

chmod +x get_build.sh

Note: On Windows, you can run the script using Git Bash.

The build files will be in the build_files.zip file. Now you can create a release on GitHub and attach the build_files.zip file.

Development process

Add a new menu

Menus structure

Menus are defined in screen_module_menu_t in the screens_module.h file. Most of them have a tree structure, you can go from one menu to another, and go back. For example in the MENU_MAIN menu, you can go to MENU_APPLICATIONS, MENU_SETTINGS, or MENU_ABOUT and go back to MENU_MAIN.

graph TD
  A[MENU_MAIN] --> B[MENU_APPLICATIONS]
  A --> C[MENU_SETTINGS]
  A --> D[MENU_ABOUT]
  B --> A
  C --> A
  D --> A
Loading

Menus can only have two possible actions: go to another menu or go back.

Let's say you want to add a new menu called MENU_GPS_WARDRIVING that will be a child of MENU_GPS wich is a child of MENU_APPLICATIONS.

graph TD
  A[MENU_MAIN] --> B[MENU_APPLICATIONS]
  B --> C[MENU_GPS]
  C --> D[MENU_GPS_WARDRIVING]
  D --> C
  B --> A
  C --> B
Loading

Steps to add a new menu

  1. The first step is to add an enum value in screen_module_menu_t, it can be anywhere, but it's recommended to place it near to it's parent to make it easier to find. In this case, we will add it after MENU_GPS.
/**
 * @brief Enum of menus
 *
 * Used to navigate through the different menus
 *
 * Modify this menu also requires to modify the `menu_list`, `next_menu_table`,
 * `prev_menu_table` and `menu_items` arrays
 */
typedef enum {
  MENU_MAIN = 0,
  MENU_APPLICATIONS,
  MENU_SETTINGS,
  MENU_ABOUT,
  /* Applications */
  ...
  MENU_GPS,
  /* GPS applications */
  MENU_GPS_WARDRIVING,
  MENU_GPS_DATE_TIME,
  MENU_GPS_LOCATION,
  MENU_GPS_SPEED,
  MENU_GPS_HELP,
  ...
  /* Menu count */
  MENU_COUNT,
} screen_module_menu_t;

Note: The MENU_COUNT is used to get the number of menus, it's important to keep it at the end of the enum because it's used to do some runtime checks.

  1. Add MENU_GPS_WARDRIVING as a string in menu_list, this is to debug purposes.
/**
 * @brief List of menus
 *
 * Used to get the menu name from the enum value
 * following the order of the `screen_module_menu_t` enum
 *
 * Usage: menu_list[screen_module_menu_t]
 */
const char* menu_list[] = {
  "MENU_MAIN",
  "MENU_APPLICATIONS",
  "MENU_SETTINGS",
  "MENU_ABOUT",
  ...
  "MENU_GPS",
  ...
  "MENU_GPS_WARDRIVING",  // Add this line
  "MENU_GPS_DATE_TIME",
  "MENU_GPS_LOCATION",
  "MENU_GPS_SPEED",
  "MENU_GPS_HELP",
  ...
};
  1. Add the option to the gps items array.
char* gps_items[] = {
    "Wardrive",  // Add this line
    "Date & Time",
    "Location",
    "Speed",
    "Help",
    NULL,
};

Note: The NULL at the end is to indicate the end of the array. It's important to add it, otherwise, the program will have undefined behavior.

  1. Define the next menus of MENU_GPS, to go to MENU_GPS_WARDRIVING in the screen_module_menu_t array. We decided to place it as the first child of MENU_GPS, this must be consistent with the order of gps_items.
/**
 * @brief List of menus
 *
 * Used to get the next menu to display when the user selects an option
 * following the order of the `screen_module_menu_t` enum
 *
 * Usage: next_menu_table[screen_module_menu_t][selected_item]
 */
const int next_menu_table[][MAX_NUM_ITEMS] = {
    // MENU_MAIN
    {MENU_APPLICATIONS, MENU_SETTINGS, MENU_ABOUT},
    ...
    // MENU_GPS
    {MENU_GPS_WARDRIVING, MENU_GPS_DATE_TIME, MENU_GPS_LOCATION, MENU_GPS_SPEED,
     MENU_GPS_HELP},
    ...
};

In this case, the comments // MENU_MAIN and // MENU_GPS represent the parent menu, they must be in the same order as the screen_module_menu_t enum. The next_menu_table is a 2D array where the first index is the parent menu and the second index is the child menu. The order of the child menus must be the same as the order of the items in the menu_items array.

  1. Define the next menu of MENU_GPS_WARDRIVING. In this case, it's an application that doesn't have any child menus, so pressing the select button should do nothing.
const int next_menu_table[][MAX_NUM_ITEMS] = {
    ...
     // MENU_GPS_WARDRIVING
    {MENU_GPS_WARDRIVING},  // Add it as itself does nothing
    ...
};
  1. Define the previous menu of MENU_GPS_WARDRIVING in the prev_menu_table array.
/**
 * @brief List of menus
 *
 * Used to get the previous menu to display when the user returns to the
 * previous menu in `menu_screens_exit_submenu`. Add the previous menu
 * following the order of the `screen_module_menu_t` enum
 *
 * Usage: prev_menu_table[screen_module_menu_t]
 */
const int prev_menu_table[] = {
    // PREVIOUS_MENU,                // CURRENT_MENU
    /*****************************************************************/
    MENU_MAIN,                       // MENU_MAIN
    MENU_MAIN,                       // MENU_APPLICATIONS
    MENU_MAIN,                       // MENU_SETTINGS
    MENU_MAIN,                       // MENU_ABOUT
    ...
    MENU_GPS,                        // MENU_GPS_WARDRIVING  --> Add this line
    MENU_GPS,                        // MENU_GPS_DATE_TIME
    MENU_GPS,                        // MENU_GPS_LOCATION
    MENU_GPS,                        // MENU_GPS_SPEED
    MENU_GPS,                        // MENU_GPS_HELP
    ...
};

Again, the comments represent the parent menu, and they must be in the same order as the screen_module_menu_t enum. The prev_menu_table is a 1D array where the index is the child menu and the value is the parent menu.

  1. Add the menu items to the menu_items array. For now, we will add an empty array, but you can add the items you want to display in the menu.
/**
 * @brief List of menu items
 *
 * Used to get the menu items from the menu enum value
 * following the order of the `screen_module_menu_t` enum
 *
 * Usage: menu_items[screen_module_menu_t]
 */
char** menu_items[] = {
    main_items,                       // MENU_MAIN
    applications_items,               // MENU_APPLICATIONS
    settings_items,                   // MENU_SETTINGS
    about_items,                      // MENU_ABOUT
    ...
    gps_items,                        // MENU_GPS
    ...
    empty_items,                      // MENU_GPS_WARDRIVING
    gps_date_time_items,              // MENU_GPS_DATE_TIME
    gps_location_items,               // MENU_GPS_LOCATION
    gps_speed_items,                  // MENU_GPS_SPEED
    gps_help,                         // MENU_GPS_HELP
    ...
};

And that's it! You should be able to navigate to the new menu. Test it and review this guide again in case you missed something.