1. ভূমিকা
Nest দ্বারা প্রকাশিত OpenThread হল Thread® নেটওয়ার্কিং প্রোটোকলের একটি ওপেন-সোর্স বাস্তবায়ন। সংযুক্ত বাড়ির জন্য পণ্যগুলির বিকাশকে ত্বরান্বিত করতে নেস্ট পণ্যগুলিতে ব্যবহৃত প্রযুক্তি বিকাশকারীদের কাছে বিস্তৃতভাবে উপলব্ধ করার জন্য নেস্ট OpenThread প্রকাশ করেছে৷
থ্রেড স্পেসিফিকেশন হোম অ্যাপ্লিকেশনের জন্য একটি IPv6-ভিত্তিক নির্ভরযোগ্য, নিরাপদ এবং কম-পাওয়ার ওয়্যারলেস ডিভাইস-টু-ডিভাইস যোগাযোগ প্রোটোকল সংজ্ঞায়িত করে। OpenThread সমস্ত থ্রেড নেটওয়ার্কিং স্তর প্রয়োগ করে যার মধ্যে IPv6, 6LoWPAN, IEEE 802.15.4 MAC নিরাপত্তা, Mesh Link Establishment, এবং Mesh Routing সহ।
এই কোডল্যাবে, আপনি একটি থ্রেড নেটওয়ার্ক শুরু করতে ওপেন থ্রেড এপিআই ব্যবহার করবেন, ডিভাইসের ভূমিকার পরিবর্তনের উপর নজর রাখতে এবং প্রতিক্রিয়া জানাতে এবং UDP বার্তা পাঠাতে, সেইসাথে বাস্তব হার্ডওয়্যারে বোতাম এবং LED-এর সাথে এই ক্রিয়াগুলিকে টাই করতে পারেন।
আপনি কি শিখবেন
- নর্ডিক nRF52840 ডেভ বোর্ডে বোতাম এবং এলইডি কীভাবে প্রোগ্রাম করবেন
- কিভাবে সাধারণ OpenThread API এবং
otInstance
ক্লাস ব্যবহার করবেন - কিভাবে ওপেন থ্রেড অবস্থার পরিবর্তনগুলি নিরীক্ষণ এবং প্রতিক্রিয়া জানাতে হয়
- কিভাবে একটি থ্রেড নেটওয়ার্কের সমস্ত ডিভাইসে UDP বার্তা পাঠাতে হয়
- মেকফাইলগুলি কীভাবে পরিবর্তন করবেন
আপনি কি প্রয়োজন হবে
হার্ডওয়্যার:
- 3 নর্ডিক সেমিকন্ডাক্টর nRF52840 ডেভ বোর্ড
- বোর্ডগুলিকে সংযুক্ত করতে 3 USB থেকে মাইক্রো-USB কেবল
- কমপক্ষে 3টি ইউএসবি পোর্ট সহ একটি লিনাক্স মেশিন
সফটওয়্যার:
- GNU টুলচেইন
- নর্ডিক nRF5x কমান্ড লাইন টুল
- সেগার জে-লিংক সফটওয়্যার
- ওপেন থ্রেড
- গিট
অন্যথায় উল্লেখ করা ছাড়া, এই কোডল্যাবের বিষয়বস্তু ক্রিয়েটিভ কমন্স অ্যাট্রিবিউশন 3.0 লাইসেন্সের অধীনে লাইসেন্সপ্রাপ্ত, এবং কোড নমুনাগুলি Apache 2.0 লাইসেন্সের অধীনে লাইসেন্সপ্রাপ্ত ।
2. শুরু করা
হার্ডওয়্যার কোডল্যাব সম্পূর্ণ করুন
এই কোডল্যাব শুরু করার আগে, আপনাকে nRF52840 বোর্ড এবং ওপেন থ্রেড কোডল্যাব দিয়ে একটি থ্রেড নেটওয়ার্ক তৈরি করতে হবে, যা:
- বিল্ডিং এবং ফ্ল্যাশ করার জন্য আপনার প্রয়োজনীয় সমস্ত সফ্টওয়্যার বিশদ বিবরণ
- আপনাকে শেখায় কিভাবে OpenThread তৈরি করতে হয় এবং Nordic nRF52840 বোর্ডে ফ্ল্যাশ করতে হয়
- একটি থ্রেড নেটওয়ার্কের বুনিয়াদি প্রদর্শন করে
ওপেন থ্রেড তৈরি করতে এবং বোর্ডগুলি ফ্ল্যাশ করার জন্য প্রয়োজনীয় পরিবেশগুলির কোনওটিই এই কোডল্যাবে বিশদ বিবরণ নেই - বোর্ডগুলি ফ্ল্যাশ করার জন্য শুধুমাত্র প্রাথমিক নির্দেশাবলী। মনে করা হচ্ছে আপনি ইতিমধ্যেই বিল্ড একটি থ্রেড নেটওয়ার্ক কোডল্যাব সম্পূর্ণ করেছেন।
লিনাক্স মেশিন
এই কোডল্যাবটিকে একটি i386- বা x86-ভিত্তিক লিনাক্স মেশিন ব্যবহার করার জন্য ডিজাইন করা হয়েছে যাতে সমস্ত থ্রেড ডেভেলপমেন্ট বোর্ড ফ্ল্যাশ করা যায়। সমস্ত পদক্ষেপ উবুন্টু 14.04.5 LTS (বিশ্বস্ত তাহর) এ পরীক্ষা করা হয়েছিল।
নর্ডিক সেমিকন্ডাক্টর nRF52840 বোর্ড
এই কোডল্যাব তিনটি nRF52840 PDK বোর্ড ব্যবহার করে।
সফটওয়্যার ইনস্টল করুন
OpenThread তৈরি এবং ফ্ল্যাশ করতে, আপনাকে SEGGER J-Link, nRF5x কমান্ড লাইন টুলস, ARM GNU টুলচেইন এবং বিভিন্ন লিনাক্স প্যাকেজ ইনস্টল করতে হবে। আপনি যদি প্রয়োজন অনুসারে একটি থ্রেড নেটওয়ার্ক কোডল্যাব তৈরি করে থাকেন তবে আপনার কাছে ইতিমধ্যেই আপনার প্রয়োজনীয় সমস্ত কিছু ইনস্টল করা থাকবে। যদি তা না হয়, তাহলে আপনি nRF52840 ডেভ বোর্ডে OpenThread তৈরি এবং ফ্ল্যাশ করতে পারেন তা নিশ্চিত করার আগে কোডল্যাবটি সম্পূর্ণ করুন।
3. ভান্ডার ক্লোন করুন
OpenThread উদাহরণ অ্যাপ্লিকেশন কোড সহ আসে যা আপনি এই কোডল্যাবের জন্য একটি সূচনা পয়েন্ট হিসাবে ব্যবহার করতে পারেন।
OpenThread নর্ডিক nRF528xx উদাহরণ রেপো ক্লোন করুন এবং OpenThread তৈরি করুন:
$ git clone --recursive https://github.com/openthread/ot-nrf528xx $ cd ot-nrf528xx $ ./script/bootstrap
4. OpenThread API বেসিক
OpenThread-এর সর্বজনীন API গুলি OpenThread সংগ্রহস্থলে ./openthread/include/openthread
এ অবস্থিত। এই APIগুলি আপনার অ্যাপ্লিকেশানগুলিতে ব্যবহারের জন্য একটি থ্রেড- এবং প্ল্যাটফর্ম-স্তরে উভয়েরই বিভিন্ন OpenThread বৈশিষ্ট্য এবং কার্যকারিতাগুলিতে অ্যাক্সেস প্রদান করে:
- OpenThread ইনস্ট্যান্স তথ্য এবং নিয়ন্ত্রণ
- অ্যাপ্লিকেশন পরিষেবাগুলি যেমন IPv6, UDP, এবং CoAP৷
- কমিশনার এবং যোগদানকারীর ভূমিকা সহ নেটওয়ার্ক শংসাপত্র ব্যবস্থাপনা
- বর্ডার রাউটার ব্যবস্থাপনা
- উন্নত বৈশিষ্ট্য যেমন শিশু তত্ত্বাবধান এবং জ্যাম সনাক্তকরণ
সমস্ত OpenThread API-এর রেফারেন্স তথ্য openthread.io/reference- এ উপলব্ধ।
একটি API ব্যবহার করে
একটি API ব্যবহার করতে, আপনার অ্যাপ্লিকেশন ফাইলগুলির একটিতে এর শিরোনাম ফাইল অন্তর্ভুক্ত করুন। তারপর পছন্দসই ফাংশন কল করুন।
উদাহরণস্বরূপ, OpenThread এর সাথে অন্তর্ভুক্ত CLI উদাহরণ অ্যাপটি নিম্নলিখিত API হেডার ব্যবহার করে:
./openthread/examples/apps/cli/main.c
#include <openthread/config.h> #include <openthread/cli.h> #include <openthread/diag.h> #include <openthread/tasklet.h> #include <openthread/platform/logging.h>
OpenThread উদাহরণ
otInstance
গঠন এমন কিছু যা আপনি OpenThread API-এর সাথে কাজ করার সময় ঘন ঘন ব্যবহার করবেন। একবার শুরু হলে, এই কাঠামোটি OpenThread লাইব্রেরির একটি স্থির উদাহরণ উপস্থাপন করে এবং ব্যবহারকারীকে OpenThread API কল করার অনুমতি দেয়।
উদাহরণস্বরূপ, OpenThread উদাহরণটি CLI উদাহরণ অ্যাপের main()
ফাংশনে আরম্ভ করা হয়েছে:
./openthread/examples/apps/cli/main.c
int main(int argc, char *argv[]) { otInstance *instance ... #if OPENTHREAD_ENABLE_MULTIPLE_INSTANCES // Call to query the buffer size (void)otInstanceInit(NULL, &otInstanceBufferLength); // Call to allocate the buffer otInstanceBuffer = (uint8_t *)malloc(otInstanceBufferLength); assert(otInstanceBuffer); // Initialize OpenThread with the buffer instance = otInstanceInit(otInstanceBuffer, &otInstanceBufferLength); #else instance = otInstanceInitSingle(); #endif ... return 0; }
প্ল্যাটফর্ম-নির্দিষ্ট ফাংশন
আপনি যদি OpenThread-এর সাথে অন্তর্ভুক্ত উদাহরণ অ্যাপ্লিকেশনগুলির একটিতে প্ল্যাটফর্ম-নির্দিষ্ট ফাংশন যোগ করতে চান, তাহলে প্রথমে সেগুলিকে ./openthread/examples/platforms/openthread-system.h
হেডারে ঘোষণা করুন, সমস্ত ফাংশনের জন্য otSys
নামস্থান ব্যবহার করে। তারপরে একটি প্ল্যাটফর্ম-নির্দিষ্ট উত্স ফাইলে তাদের বাস্তবায়ন করুন। এইভাবে বিমূর্ত, আপনি অন্যান্য উদাহরণ প্ল্যাটফর্মের জন্য একই ফাংশন হেডার ব্যবহার করতে পারেন।
উদাহরণ স্বরূপ, আমরা nRF52840 বোতাম এবং LED তে হুক করার জন্য যে GPIO ফাংশনগুলি ব্যবহার করতে যাচ্ছি openthread-system.h
এ ঘোষণা করতে হবে।
আপনার পছন্দের টেক্সট এডিটরে ./openthread/examples/platforms/openthread-system.h
ফাইলটি খুলুন।
./openthread/examples/platforms/openthread-system.h
অ্যাকশন: প্ল্যাটফর্ম-নির্দিষ্ট GPIO ফাংশন ঘোষণা যোগ করুন।
openthread/instance.h
হেডারের জন্য #include
এর পরে এই ফাংশন ঘোষণা যোগ করুন:
/** * Init LED module. * */ void otSysLedInit(void); void otSysLedSet(uint8_t aLed, bool aOn); void otSysLedToggle(uint8_t aLed); /** * A callback will be called when GPIO interrupts occur. * */ typedef void (*otSysButtonCallback)(otInstance *aInstance); void otSysButtonInit(otSysButtonCallback aCallback); void otSysButtonProcess(otInstance *aInstance);
আমরা পরবর্তী ধাপে এগুলো বাস্তবায়ন করব।
মনে রাখবেন যে otSysButtonProcess
ফাংশন ঘোষণা একটি otInstance
ব্যবহার করে। এইভাবে অ্যাপ্লিকেশনটি OpenThread ইনস্ট্যান্স সম্পর্কে তথ্য অ্যাক্সেস করতে পারে যখন প্রয়োজন হলে একটি বোতাম টিপানো হয়। এটি সব আপনার আবেদনের প্রয়োজনের উপর নির্ভর করে। আপনার যদি ফাংশনটি বাস্তবায়নের জন্য এটির প্রয়োজন না হয়, আপনি কিছু টুলচেইনের জন্য অব্যবহৃত ভেরিয়েবলের চারপাশে বিল্ড ত্রুটিগুলি দমন করতে OpenThread API থেকে OT_UNUSED_VARIABLE
ম্যাক্রো ব্যবহার করতে পারেন। এর উদাহরণ আমরা পরে দেখব।
5. GPIO প্ল্যাটফর্ম বিমূর্ততা বাস্তবায়ন করুন
পূর্ববর্তী ধাপে, আমরা ./openthread/examples/platforms/openthread-system.h
এ প্ল্যাটফর্ম-নির্দিষ্ট ফাংশন ঘোষণাগুলি দেখেছি যা GPIO-এর জন্য ব্যবহার করা যেতে পারে। nRF52840 ডেভ বোর্ডগুলিতে বোতাম এবং LED অ্যাক্সেস করার জন্য, আপনাকে nRF52840 প্ল্যাটফর্মের জন্য সেই ফাংশনগুলি বাস্তবায়ন করতে হবে। এই কোডে, আপনি ফাংশন যোগ করবেন যা:
- GPIO পিন এবং মোড শুরু করুন
- একটি পিনে ভোল্টেজ নিয়ন্ত্রণ করুন
- GPIO বাধাগুলি সক্ষম করুন এবং একটি কলব্যাক নিবন্ধন করুন৷
./src/src
ডিরেক্টরিতে, gpio.c
নামে একটি নতুন ফাইল তৈরি করুন। এই নতুন ফাইলে, নিম্নলিখিত বিষয়বস্তু যোগ করুন.
./src/src/gpio.c (নতুন ফাইল)
অ্যাকশন: সংজ্ঞায়িত যোগ করুন।
এই সংজ্ঞাগুলি nRF52840-নির্দিষ্ট মান এবং OpenThread অ্যাপ্লিকেশন স্তরে ব্যবহৃত ভেরিয়েবলগুলির মধ্যে বিমূর্ততা হিসাবে কাজ করে।
/** * @file * This file implements the system abstraction for GPIO and GPIOTE. * */ #define BUTTON_GPIO_PORT 0x50000300UL #define BUTTON_PIN 11 // button #1 #define GPIO_LOGIC_HI 0 #define GPIO_LOGIC_LOW 1 #define LED_GPIO_PORT 0x50000300UL #define LED_1_PIN 13 // turn on to indicate leader role #define LED_2_PIN 14 // turn on to indicate router role #define LED_3_PIN 15 // turn on to indicate child role #define LED_4_PIN 16 // turn on to indicate UDP receive
nRF52840 বোতাম এবং LED সম্পর্কে আরও তথ্যের জন্য, নর্ডিক সেমিকন্ডাক্টর ইনফোসেন্টার দেখুন।
অ্যাকশন: শিরোনাম যোগ করুন অন্তর্ভুক্ত.
এরপরে, GPIO কার্যকারিতার জন্য আপনার প্রয়োজন হবে এমন শিরোনাম যোগ করুন।
/* Header for the functions defined here */ #include "openthread-system.h" #include <string.h> /* Header to access an OpenThread instance */ #include <openthread/instance.h> /* Headers for lower-level nRF52840 functions */ #include "platform-nrf5.h" #include "hal/nrf_gpio.h" #include "hal/nrf_gpiote.h" #include "nrfx/drivers/include/nrfx_gpiote.h"
অ্যাকশন: বোতাম 1 এর জন্য কলব্যাক এবং বাধা ফাংশন যোগ করুন।
পরবর্তী এই কোড যোগ করুন. in_pin1_handler
ফাংশন হল কলব্যাক যা নিবন্ধিত হয় যখন বোতাম প্রেস কার্যকারিতা শুরু হয় (পরে এই ফাইলে)।
নোট করুন কিভাবে এই কলব্যাকটি OT_UNUSED_VARIABLE
ম্যাক্রো ব্যবহার করে, কারণ in_pin1_handler
এ পাস করা ভেরিয়েবল আসলে ফাংশনে ব্যবহার করা হয় না।
/* Declaring callback function for button 1. */ static otSysButtonCallback sButtonHandler; static bool sButtonPressed; /** * @brief Function to receive interrupt and call back function * set by the application for button 1. * */ static void in_pin1_handler(uint32_t pin, nrf_gpiote_polarity_t action) { OT_UNUSED_VARIABLE(pin); OT_UNUSED_VARIABLE(action); sButtonPressed = true; }
অ্যাকশন: এলইডি কনফিগার করতে একটি ফাংশন যোগ করুন।
আরম্ভ করার সময় সমস্ত LED-এর মোড এবং অবস্থা কনফিগার করতে এই কোড যোগ করুন।
/** * @brief Function for configuring: PIN_IN pin for input, PIN_OUT pin for output, * and configures GPIOTE to give an interrupt on pin change. */ void otSysLedInit(void) { /* Configure GPIO mode: output */ nrf_gpio_cfg_output(LED_1_PIN); nrf_gpio_cfg_output(LED_2_PIN); nrf_gpio_cfg_output(LED_3_PIN); nrf_gpio_cfg_output(LED_4_PIN); /* Clear all output first */ nrf_gpio_pin_write(LED_1_PIN, GPIO_LOGIC_LOW); nrf_gpio_pin_write(LED_2_PIN, GPIO_LOGIC_LOW); nrf_gpio_pin_write(LED_3_PIN, GPIO_LOGIC_LOW); nrf_gpio_pin_write(LED_4_PIN, GPIO_LOGIC_LOW); /* Initialize gpiote for button(s) input. Button event handlers are set in the application (main.c) */ ret_code_t err_code; err_code = nrfx_gpiote_init(); APP_ERROR_CHECK(err_code); }
অ্যাকশন: একটি LED এর মোড সেট করতে একটি ফাংশন যোগ করুন।
ডিভাইসের ভূমিকা পরিবর্তন হলে এই ফাংশনটি ব্যবহার করা হবে।
/** * @brief Function to set the mode of an LED. */ void otSysLedSet(uint8_t aLed, bool aOn) { switch (aLed) { case 1: nrf_gpio_pin_write(LED_1_PIN, (aOn == GPIO_LOGIC_HI)); break; case 2: nrf_gpio_pin_write(LED_2_PIN, (aOn == GPIO_LOGIC_HI)); break; case 3: nrf_gpio_pin_write(LED_3_PIN, (aOn == GPIO_LOGIC_HI)); break; case 4: nrf_gpio_pin_write(LED_4_PIN, (aOn == GPIO_LOGIC_HI)); break; } }
অ্যাকশন: একটি LED এর মোড টগল করতে একটি ফাংশন যোগ করুন।
যখন ডিভাইসটি একটি মাল্টিকাস্ট UDP বার্তা পায় তখন এই ফাংশনটি LED4 টগল করতে ব্যবহার করা হবে।
/** * @brief Function to toggle the mode of an LED. */ void otSysLedToggle(uint8_t aLed) { switch (aLed) { case 1: nrf_gpio_pin_toggle(LED_1_PIN); break; case 2: nrf_gpio_pin_toggle(LED_2_PIN); break; case 3: nrf_gpio_pin_toggle(LED_3_PIN); break; case 4: nrf_gpio_pin_toggle(LED_4_PIN); break; } }
অ্যাকশন: বোতাম টিপে শুরু এবং প্রক্রিয়া করার জন্য ফাংশন যোগ করুন।
প্রথম ফাংশনটি একটি বোতাম প্রেসের জন্য বোর্ডকে আরম্ভ করে, এবং দ্বিতীয়টি মাল্টিকাস্ট UDP বার্তা পাঠায় যখন বোতাম 1 টিপানো হয়।
/** * @brief Function to initialize the button. */ void otSysButtonInit(otSysButtonCallback aCallback) { nrfx_gpiote_in_config_t in_config = NRFX_GPIOTE_CONFIG_IN_SENSE_LOTOHI(true); in_config.pull = NRF_GPIO_PIN_PULLUP; ret_code_t err_code; err_code = nrfx_gpiote_in_init(BUTTON_PIN, &in_config, in_pin1_handler); APP_ERROR_CHECK(err_code); sButtonHandler = aCallback; sButtonPressed = false; nrfx_gpiote_in_event_enable(BUTTON_PIN, true); } void otSysButtonProcess(otInstance *aInstance) { if (sButtonPressed) { sButtonPressed = false; sButtonHandler(aInstance); } }
অ্যাকশন: gpio.c
ফাইলটি সংরক্ষণ করুন এবং বন্ধ করুন।
6. API: ডিভাইসের ভূমিকা পরিবর্তনের প্রতিক্রিয়া
আমাদের অ্যাপ্লিকেশনে, আমরা চাই ডিভাইসের ভূমিকার উপর নির্ভর করে বিভিন্ন LED আলোকিত হোক। আসুন নিম্নলিখিত ভূমিকাগুলি ট্র্যাক করি: লিডার, রাউটার, এন্ড ডিভাইস। আমরা সেগুলিকে এলইডিতে বরাদ্দ করতে পারি এভাবে:
- LED1 = নেতা
- LED2 = রাউটার
- LED3 = শেষ ডিভাইস
এই কার্যকারিতা সক্ষম করার জন্য, অ্যাপ্লিকেশনটিকে জানতে হবে কখন ডিভাইসের ভূমিকা পরিবর্তিত হয়েছে এবং প্রতিক্রিয়াতে কীভাবে সঠিক LED চালু করতে হবে। আমরা প্রথম অংশের জন্য OpenThread উদাহরণ এবং দ্বিতীয় অংশের জন্য GPIO প্ল্যাটফর্ম বিমূর্ততা ব্যবহার করব।
আপনার পছন্দের টেক্সট এডিটরে ./openthread/examples/apps/cli/main.c
ফাইলটি খুলুন।
./openthread/examples/apps/cli/main.c
অ্যাকশন: শিরোনাম যোগ করুন অন্তর্ভুক্ত.
main.c
ফাইলের অন্তর্ভুক্ত বিভাগে, ভূমিকা পরিবর্তন বৈশিষ্ট্যের জন্য আপনার প্রয়োজন হবে এমন API শিরোনাম ফাইল যোগ করুন।
#include <openthread/instance.h> #include <openthread/thread.h> #include <openthread/thread_ftd.h>
অ্যাকশন: OpenThread ইনস্ট্যান্স স্টেট পরিবর্তনের জন্য হ্যান্ডলার ফাংশন ঘোষণা যোগ করুন।
এই ঘোষণাটি main.c
এ যুক্ত করুন, শিরোনামটি অন্তর্ভুক্ত করার পরে এবং যেকোনো #if
বিবৃতির আগে। এই ফাংশন প্রধান অ্যাপ্লিকেশন পরে সংজ্ঞায়িত করা হবে.
void handleNetifStateChanged(uint32_t aFlags, void *aContext);
অ্যাকশন: রাজ্য পরিবর্তন হ্যান্ডলার ফাংশনের জন্য একটি কলব্যাক নিবন্ধন যোগ করুন।
main.c
এ, otAppCliInit
কলের পরে main()
ফাংশনে এই ফাংশনটি যোগ করুন। এই কলব্যাক রেজিস্ট্রেশনটি OpenThread কে handleNetifStateChange
ফাংশন কল করতে বলে যখনই OpenThread ইনস্ট্যান্সের অবস্থা পরিবর্তন হয়।
/* Register Thread state change handler */ otSetStateChangedCallback(instance, handleNetifStateChanged, instance);
অ্যাকশন: রাষ্ট্র পরিবর্তন বাস্তবায়ন যোগ করুন.
main.c
এ, main()
ফাংশনের পরে, handleNetifStateChanged
ফাংশনটি প্রয়োগ করুন। এই ফাংশনটি OpenThread ইনস্ট্যান্সের OT_CHANGED_THREAD_ROLE
পতাকা পরীক্ষা করে এবং যদি এটি পরিবর্তিত হয়ে থাকে, প্রয়োজন অনুযায়ী LED চালু/বন্ধ করে।
void handleNetifStateChanged(uint32_t aFlags, void *aContext) { if ((aFlags & OT_CHANGED_THREAD_ROLE) != 0) { otDeviceRole changedRole = otThreadGetDeviceRole(aContext); switch (changedRole) { case OT_DEVICE_ROLE_LEADER: otSysLedSet(1, true); otSysLedSet(2, false); otSysLedSet(3, false); break; case OT_DEVICE_ROLE_ROUTER: otSysLedSet(1, false); otSysLedSet(2, true); otSysLedSet(3, false); break; case OT_DEVICE_ROLE_CHILD: otSysLedSet(1, false); otSysLedSet(2, false); otSysLedSet(3, true); break; case OT_DEVICE_ROLE_DETACHED: case OT_DEVICE_ROLE_DISABLED: /* Clear LED4 if Thread is not enabled. */ otSysLedSet(4, false); break; } } }
7. API: একটি LED চালু করতে মাল্টিকাস্ট ব্যবহার করুন
আমাদের অ্যাপ্লিকেশনে, যখন একটি বোর্ডে বোতাম 1 টিপানো হয় তখন আমরা নেটওয়ার্কের অন্যান্য সমস্ত ডিভাইসে UDP বার্তা পাঠাতে চাই। বার্তার প্রাপ্তি নিশ্চিত করতে, আমরা প্রতিক্রিয়া হিসাবে অন্যান্য বোর্ডগুলিতে LED4 টগল করব৷
এই কার্যকারিতা সক্ষম করতে, অ্যাপ্লিকেশনটির প্রয়োজন:
- শুরু করার পরে একটি UDP সংযোগ শুরু করুন
- মেশ-স্থানীয় মাল্টিকাস্ট ঠিকানায় একটি UDP বার্তা পাঠাতে সক্ষম হন
- ইনকামিং UDP বার্তা পরিচালনা করুন
- আগত UDP বার্তাগুলির প্রতিক্রিয়া হিসাবে LED4 টগল করুন
আপনার পছন্দের টেক্সট এডিটরে ./openthread/examples/apps/cli/main.c
ফাইলটি খুলুন।
./openthread/examples/apps/cli/main.c
অ্যাকশন: শিরোনাম যোগ করুন অন্তর্ভুক্ত.
main.c
ফাইলের শীর্ষে অন্তর্ভুক্ত বিভাগে, মাল্টিকাস্ট UDP বৈশিষ্ট্যের জন্য আপনার প্রয়োজনীয় API শিরোনাম ফাইলগুলি যোগ করুন।
#include <string.h> #include <openthread/message.h> #include <openthread/udp.h> #include "utils/code_utils.h"
code_utils.h
শিরোনামটি otEXPECT
এবং otEXPECT_ACTION
ম্যাক্রোগুলির জন্য ব্যবহৃত হয় যা রান-টাইম শর্তগুলিকে যাচাই করে এবং ত্রুটিগুলি সুন্দরভাবে পরিচালনা করে৷
অ্যাকশন: সংজ্ঞা এবং ধ্রুবক যোগ করুন:
main.c
ফাইলে, অন্তর্ভুক্ত বিভাগের পরে এবং যেকোনো #if
স্টেটমেন্টের আগে, UDP-নির্দিষ্ট ধ্রুবক যোগ করুন এবং সংজ্ঞায়িত করুন:
#define UDP_PORT 1212 static const char UDP_DEST_ADDR[] = "ff03::1"; static const char UDP_PAYLOAD[] = "Hello OpenThread World!";
ff03::1
হল জাল-স্থানীয় মাল্টিকাস্ট ঠিকানা। এই ঠিকানায় প্রেরিত যেকোনো বার্তা নেটওয়ার্কের সমস্ত সম্পূর্ণ থ্রেড ডিভাইসে পাঠানো হবে। OpenThread-এ মাল্টিকাস্ট সমর্থন সম্পর্কে আরও তথ্যের জন্য openthread.io-তে মাল্টিকাস্ট দেখুন।
অ্যাকশন: ফাংশন ঘোষণা যোগ করুন।
main.c
ফাইলে, otTaskletsSignalPending
সংজ্ঞার পরে এবং main()
ফাংশনের আগে, UDP-নির্দিষ্ট ফাংশন যোগ করুন, সেইসাথে একটি UDP সকেট উপস্থাপন করার জন্য একটি স্ট্যাটিক ভেরিয়েবল:
static void initUdp(otInstance *aInstance); static void sendUdp(otInstance *aInstance); static void handleButtonInterrupt(otInstance *aInstance); void handleUdpReceive(void *aContext, otMessage *aMessage, const otMessageInfo *aMessageInfo); static otUdpSocket sUdpSocket;
অ্যাকশন: GPIO LEDs এবং বোতাম চালু করতে কল যোগ করুন।
main.c
এ, otSetStateChangedCallback
কলের পরে main()
ফাংশনে এই ফাংশন কলগুলি যোগ করুন। এই ফাংশনগুলি GPIO এবং GPIOTE পিনগুলি শুরু করে এবং বোতাম পুশ ইভেন্টগুলি পরিচালনা করার জন্য একটি বোতাম হ্যান্ডলার সেট করে।
/* init GPIO LEDs and button */ otSysLedInit(); otSysButtonInit(handleButtonInterrupt);
অ্যাকশন: UDP ইনিশিয়ালাইজেশন কল যোগ করুন।
main.c
এ, এই ফাংশনটি main()
ফাংশনে যোগ করুন otSysButtonInit
কলের পরে আপনি এইমাত্র যোগ করেছেন:
initUdp(instance);
এই কলটি নিশ্চিত করে যে অ্যাপ্লিকেশন শুরু হওয়ার পরে একটি UDP সকেট শুরু হয়েছে। এটি ছাড়া, ডিভাইসটি UDP বার্তা পাঠাতে বা গ্রহণ করতে পারে না।
অ্যাকশন: GPIO বোতাম ইভেন্ট প্রক্রিয়া করতে কল যোগ করুন।
main.c
এ, এই ফাংশন কলটি main()
ফাংশনে যুক্ত করুন otSysProcessDrivers
কলের পরে, while
লুপে। এই ফাংশনটি, gpio.c
এ ঘোষিত, বোতাম টিপছে কিনা তা পরীক্ষা করে, এবং যদি তাই হয়, হ্যান্ডলারকে কল করে ( handleButtonInterrupt
) যা উপরের ধাপে সেট করা হয়েছিল।
otSysButtonProcess(instance);
অ্যাকশন: বোতাম ইন্টারাপ্ট হ্যান্ডলার প্রয়োগ করুন।
main.c
তে, আপনি আগের ধাপে যোগ করা handleNetifStateChanged
ফাংশনের পরে handleButtonInterrupt
ফাংশনের বাস্তবায়ন যোগ করুন:
/** * Function to handle button push event */ void handleButtonInterrupt(otInstance *aInstance) { sendUdp(aInstance); }
অ্যাকশন: UDP প্রারম্ভিকতা বাস্তবায়ন করুন।
main.c
এ, আপনি এইমাত্র যোগ করা handleButtonInterrupt
ফাংশনের পরে initUdp
ফাংশন বাস্তবায়ন যোগ করুন:
/** * Initialize UDP socket */ void initUdp(otInstance *aInstance) { otSockAddr listenSockAddr; memset(&sUdpSocket, 0, sizeof(sUdpSocket)); memset(&listenSockAddr, 0, sizeof(listenSockAddr)); listenSockAddr.mPort = UDP_PORT; otUdpOpen(aInstance, &sUdpSocket, handleUdpReceive, aInstance); otUdpBind(aInstance, &sUdpSocket, &listenSockAddr, OT_NETIF_THREAD); }
UDP_PORT
হল সেই পোর্ট যা আপনি আগে সংজ্ঞায়িত করেছেন (1212)। otUdpOpen
ফাংশন সকেটটি খোলে এবং একটি UDP বার্তা প্রাপ্ত হওয়ার জন্য একটি কলব্যাক ফাংশন ( handleUdpReceive
) নিবন্ধন করে। otUdpBind
OT_NETIF_THREAD
পাস করে থ্রেড নেটওয়ার্ক ইন্টারফেসের সাথে সকেটকে আবদ্ধ করে। অন্যান্য নেটওয়ার্ক ইন্টারফেস বিকল্পের জন্য, UDP API রেফারেন্সে otNetifIdentifier
গণনা পড়ুন।
অ্যাকশন: UDP মেসেজিং প্রয়োগ করুন।
main.c
তে, আপনি এইমাত্র যোগ করা initUdp
ফাংশনের পরে sendUdp
ফাংশনের বাস্তবায়ন যোগ করুন:
/** * Send a UDP datagram */ void sendUdp(otInstance *aInstance) { otError error = OT_ERROR_NONE; otMessage * message; otMessageInfo messageInfo; otIp6Address destinationAddr; memset(&messageInfo, 0, sizeof(messageInfo)); otIp6AddressFromString(UDP_DEST_ADDR, &destinationAddr); messageInfo.mPeerAddr = destinationAddr; messageInfo.mPeerPort = UDP_PORT; message = otUdpNewMessage(aInstance, NULL); otEXPECT_ACTION(message != NULL, error = OT_ERROR_NO_BUFS); error = otMessageAppend(message, UDP_PAYLOAD, sizeof(UDP_PAYLOAD)); otEXPECT(error == OT_ERROR_NONE); error = otUdpSend(aInstance, &sUdpSocket, message, &messageInfo); exit: if (error != OT_ERROR_NONE && message != NULL) { otMessageFree(message); } }
otEXPECT
এবং otEXPECT_ACTION
ম্যাক্রো নোট করুন। এগুলি নিশ্চিত করে যে UDP বার্তাটি বৈধ এবং বাফারে সঠিকভাবে বরাদ্দ করা হয়েছে, এবং যদি তা না হয়, ফাংশনটি exit
ব্লকে ঝাঁপ দিয়ে ত্রুটিগুলি সুন্দরভাবে পরিচালনা করে, যেখানে এটি বাফারকে মুক্ত করে।
UDP আরম্ভ করার জন্য ব্যবহৃত ফাংশন সম্পর্কে আরও তথ্যের জন্য openthread.io-তে IPv6 এবং UDP রেফারেন্স দেখুন।
অ্যাকশন: UDP মেসেজ হ্যান্ডলিং বাস্তবায়ন করুন।
main.c
এ, আপনি এইমাত্র যোগ করা sendUdp
ফাংশনের পরে handleUdpReceive
ফাংশনের বাস্তবায়ন যোগ করুন। এই ফাংশনটি কেবল LED4 টগল করে।
/** * Function to handle UDP datagrams received on the listening socket */ void handleUdpReceive(void *aContext, otMessage *aMessage, const otMessageInfo *aMessageInfo) { OT_UNUSED_VARIABLE(aContext); OT_UNUSED_VARIABLE(aMessage); OT_UNUSED_VARIABLE(aMessageInfo); otSysLedToggle(4); }
8. API: থ্রেড নেটওয়ার্ক কনফিগার করুন
প্রদর্শনের স্বাচ্ছন্দ্যের জন্য, আমরা চাই আমাদের ডিভাইসগুলি অবিলম্বে থ্রেড শুরু করুক এবং একটি নেটওয়ার্কে একসাথে যোগদান করুক যখন সেগুলি চালু থাকবে। এটি করার জন্য, আমরা otOperationalDataset
কাঠামো ব্যবহার করব। এই কাঠামোটি একটি ডিভাইসে থ্রেড নেটওয়ার্ক শংসাপত্র প্রেরণের জন্য প্রয়োজনীয় সমস্ত পরামিতি ধারণ করে।
এই কাঠামোর ব্যবহার OpenThread-এ নির্মিত নেটওয়ার্ক ডিফল্টগুলিকে ওভাররাইড করবে, আমাদের অ্যাপ্লিকেশনকে আরও সুরক্ষিত করতে এবং আমাদের নেটওয়ার্কে থ্রেড নোডগুলিকে শুধুমাত্র যারা অ্যাপ্লিকেশন চালাচ্ছে তাদের মধ্যে সীমাবদ্ধ করবে।
আবার, আপনার পছন্দের টেক্সট এডিটরে ./openthread/examples/apps/cli/main.c
ফাইলটি খুলুন।
./openthread/examples/apps/cli/main.c
কর্ম: শিরোনাম যোগ করুন অন্তর্ভুক্ত.
main.c
ফাইলের শীর্ষে থাকা অন্তর্ভুক্ত বিভাগের মধ্যে, থ্রেড নেটওয়ার্ক কনফিগার করতে আপনার প্রয়োজন হবে এমন API হেডার ফাইল যোগ করুন:
#include <openthread/dataset_ftd.h>
অ্যাকশন: নেটওয়ার্ক কনফিগারেশন সেট করার জন্য ফাংশন ঘোষণা যোগ করুন।
এই ঘোষণাটি main.c
এ যুক্ত করুন, শিরোনামটি অন্তর্ভুক্ত করার পরে এবং যেকোনো #if
বিবৃতির আগে। এই ফাংশন প্রধান অ্যাপ্লিকেশন ফাংশন পরে সংজ্ঞায়িত করা হবে.
static void setNetworkConfiguration(otInstance *aInstance);
অ্যাকশন: নেটওয়ার্ক কনফিগারেশন কল যোগ করুন।
main.c
এ, otSetStateChangedCallback
কলের পরে main()
ফাংশনে এই ফাংশন কল যোগ করুন। এই ফাংশনটি থ্রেড নেটওয়ার্ক ডেটাসেট কনফিগার করে।
/* Override default network credentials */ setNetworkConfiguration(instance);
অ্যাকশন: থ্রেড নেটওয়ার্ক ইন্টারফেস এবং স্ট্যাক সক্ষম করতে কল যোগ করুন।
main.c
এ, otSysButtonInit
কলের পরে main()
ফাংশনে এই ফাংশন কল যোগ করুন।
/* Start the Thread network interface (CLI cmd > ifconfig up) */ otIp6SetEnabled(instance, true); /* Start the Thread stack (CLI cmd > thread start) */ otThreadSetEnabled(instance, true);
অ্যাকশন: থ্রেড নেটওয়ার্ক কনফিগারেশন বাস্তবায়ন করুন।
main.c
এ, main()
ফাংশনের পরে setNetworkConfiguration
ফাংশনের বাস্তবায়ন যোগ করুন:
/** * Override default network settings, such as panid, so the devices can join a network */ void setNetworkConfiguration(otInstance *aInstance) { static char aNetworkName[] = "OTCodelab"; otOperationalDataset aDataset; memset(&aDataset, 0, sizeof(otOperationalDataset)); /* * Fields that can be configured in otOperationDataset to override defaults: * Network Name, Mesh Local Prefix, Extended PAN ID, PAN ID, Delay Timer, * Channel, Channel Mask Page 0, Network Key, PSKc, Security Policy */ aDataset.mActiveTimestamp.mSeconds = 1; aDataset.mActiveTimestamp.mTicks = 0; aDataset.mActiveTimestamp.mAuthoritative = false; aDataset.mComponents.mIsActiveTimestampPresent = true; /* Set Channel to 15 */ aDataset.mChannel = 15; aDataset.mComponents.mIsChannelPresent = true; /* Set Pan ID to 2222 */ aDataset.mPanId = (otPanId)0x2222; aDataset.mComponents.mIsPanIdPresent = true; /* Set Extended Pan ID to C0DE1AB5C0DE1AB5 */ uint8_t extPanId[OT_EXT_PAN_ID_SIZE] = {0xC0, 0xDE, 0x1A, 0xB5, 0xC0, 0xDE, 0x1A, 0xB5}; memcpy(aDataset.mExtendedPanId.m8, extPanId, sizeof(aDataset.mExtendedPanId)); aDataset.mComponents.mIsExtendedPanIdPresent = true; /* Set network key to 1234C0DE1AB51234C0DE1AB51234C0DE */ uint8_t key[OT_NETWORK_KEY_SIZE] = {0x12, 0x34, 0xC0, 0xDE, 0x1A, 0xB5, 0x12, 0x34, 0xC0, 0xDE, 0x1A, 0xB5, 0x12, 0x34, 0xC0, 0xDE}; memcpy(aDataset.mNetworkKey.m8, key, sizeof(aDataset.mNetworkKey)); aDataset.mComponents.mIsNetworkKeyPresent = true; /* Set Network Name to OTCodelab */ size_t length = strlen(aNetworkName); assert(length <= OT_NETWORK_NAME_MAX_SIZE); memcpy(aDataset.mNetworkName.m8, aNetworkName, length); aDataset.mComponents.mIsNetworkNamePresent = true; otDatasetSetActive(aInstance, &aDataset); /* Set the router selection jitter to override the 2 minute default. CLI cmd > routerselectionjitter 20 Warning: For demo purposes only - not to be used in a real product */ uint8_t jitterValue = 20; otThreadSetRouterSelectionJitter(aInstance, jitterValue); }
ফাংশনে বিশদ হিসাবে, আমরা এই অ্যাপ্লিকেশনটির জন্য যে থ্রেড নেটওয়ার্ক প্যারামিটারগুলি ব্যবহার করছি তা হল:
- চ্যানেল = 15
- প্যান আইডি = 0x2222
- বর্ধিত প্যান আইডি = C0DE1AB5C0DE1AB5
- নেটওয়ার্ক কী = 1234C0DE1AB51234C0DE1AB51234C0DE
- নেটওয়ার্কের নাম = OTCodelab
উপরন্তু, এখানেই আমরা রাউটার সিলেকশন জিটার কমিয়ে ফেলি, তাই ডেমো উদ্দেশ্যে আমাদের ডিভাইসগুলি দ্রুত ভূমিকা পরিবর্তন করে। মনে রাখবেন যে নোডটি একটি FTD (ফুল থ্রেড ডিভাইস) হলেই এটি করা হয়। পরবর্তী ধাপে যে সম্পর্কে আরো.
9. API: সীমাবদ্ধ ফাংশন
OpenThread এর কিছু API সেটিংস পরিবর্তন করে যা শুধুমাত্র ডেমো বা পরীক্ষার উদ্দেশ্যে পরিবর্তন করা উচিত। এই API গুলি OpenThread ব্যবহার করে একটি অ্যাপ্লিকেশনের উৎপাদন স্থাপনায় ব্যবহার করা উচিত নয়।
উদাহরণস্বরূপ, otThreadSetRouterSelectionJitter
ফাংশন রাউটারে নিজেকে উন্নীত করতে একটি এন্ড ডিভাইসের জন্য যে সময় লাগে (সেকেন্ডে) তা সামঞ্জস্য করে। থ্রেড স্পেসিফিকেশন অনুযায়ী এই মানের জন্য ডিফল্ট হল 120। এই কোডল্যাবে ব্যবহারের সুবিধার জন্য, আমরা এটিকে 20 এ পরিবর্তন করতে যাচ্ছি, যাতে ভূমিকা পরিবর্তন করার জন্য আপনাকে থ্রেড নোডের জন্য খুব বেশি অপেক্ষা করতে হবে না।
দ্রষ্টব্য: MTD ডিভাইস রাউটারে পরিণত হয় না, এবং otThreadSetRouterSelectionJitter
এর মতো একটি ফাংশনের জন্য সমর্থন একটি MTD বিল্ডে অন্তর্ভুক্ত নয়। পরে আমাদের CMake বিকল্প -DOT_MTD=OFF
নির্দিষ্ট করতে হবে, অন্যথায় আমরা একটি বিল্ড ব্যর্থতার সম্মুখীন হব।
আপনি otThreadSetRouterSelectionJitter
ফাংশন সংজ্ঞা দেখে এটি নিশ্চিত করতে পারেন, যা OPENTHREAD_FTD
এর একটি প্রিপ্রসেসর নির্দেশের মধ্যে রয়েছে:
./openthread/src/core/api/thread_ftd_api.cpp
#if OPENTHREAD_FTD #include <openthread/thread_ftd.h> ... void otThreadSetRouterSelectionJitter(otInstance *aInstance, uint8_t aRouterJitter) { Instance &instance = *static_cast<Instance *>(aInstance); instance.GetThreadNetif().GetMle().SetRouterSelectionJitter(aRouterJitter); } ... #endif // OPENTHREAD_FTD
10. CMake আপডেট
আপনি আপনার অ্যাপ্লিকেশন তৈরি করার আগে, তিনটি CMake ফাইলের জন্য কয়েকটি ছোটখাটো আপডেট প্রয়োজন। এগুলি আপনার অ্যাপ্লিকেশনকে কম্পাইল এবং লিঙ্ক করতে বিল্ড সিস্টেম দ্বারা ব্যবহৃত হয়।
./third_party/NordicSemiconductor/CMakeLists.txt
এখন নর্ডিক সেমিকন্ডাক্টর CMakeLists.txt
এ কিছু ফ্ল্যাগ যোগ করুন, যাতে GPIO ফাংশনগুলি অ্যাপ্লিকেশানে সংজ্ঞায়িত করা হয়েছে।
অ্যাকশন: CMakeLists.txt
ফাইলে পতাকা যোগ করুন।
আপনার পছন্দের পাঠ্য সম্পাদকে ./third_party/NordicSemiconductor/CMakeLists.txt
খুলুন এবং COMMON_FLAG
বিভাগে নিম্নলিখিত লাইনগুলি যোগ করুন।
... set(COMMON_FLAG -DSPIS_ENABLED=1 -DSPIS0_ENABLED=1 -DNRFX_SPIS_ENABLED=1 -DNRFX_SPIS0_ENABLED=1 ... # Defined in ./third_party/NordicSemiconductor/nrfx/templates/nRF52840/nrfx_config.h -DGPIOTE_ENABLED=1 -DGPIOTE_CONFIG_IRQ_PRIORITY=7 -DGPIOTE_CONFIG_NUM_OF_LOW_POWER_EVENTS=1 ) ...
./src/CMakeLists.txt
নতুন gpio.c
সোর্স ফাইল যোগ করতে ./src/CMakeLists.txt
ফাইলটি সম্পাদনা করুন:
অ্যাকশন: ./src/CMakeLists.txt
ফাইলে gpio উৎস যোগ করুন ।
আপনার পছন্দের পাঠ্য সম্পাদকে ./src/CMakeLists.txt
খুলুন এবং ফাইলটি NRF_COMM_SOURCES
বিভাগে যোগ করুন।
... set(NRF_COMM_SOURCES ... src/gpio.c ... ) ...
./third_party/NordicSemiconductor/CMakeLists.txt
সবশেষে, NordicSemiconductor CMakeLists.txt
ফাইলে nrfx_gpiote.c
ড্রাইভার ফাইল যোগ করুন, যাতে এটি নর্ডিক ড্রাইভারের লাইব্রেরি বিল্ডের সাথে অন্তর্ভুক্ত করা হয়।
অ্যাকশন: NordicSemiconductor CMakeLists.txt
ফাইলে gpio ড্রাইভার যোগ করুন ।
আপনার পছন্দের পাঠ্য সম্পাদকে ./third_party/NordicSemiconductor/CMakeLists.txt
খুলুন এবং COMMON_SOURCES
বিভাগে ফাইলটি যুক্ত করুন৷
... set(COMMON_SOURCES ... nrfx/drivers/src/nrfx_gpiote.c ... ) ...
11. ডিভাইসগুলি সেট আপ করুন৷
সমস্ত কোড আপডেট হয়ে গেলে, আপনি তিনটি নর্ডিক nRF52840 ডেভ বোর্ডে অ্যাপ্লিকেশনটি তৈরি এবং ফ্ল্যাশ করতে প্রস্তুত। প্রতিটি ডিভাইস ফুল থ্রেড ডিভাইস (FTD) হিসেবে কাজ করবে।
OpenThread তৈরি করুন
nRF52840 প্ল্যাটফর্মের জন্য OpenThread FTD বাইনারি তৈরি করুন।
$ cd ~/ot-nrf528xx $ ./script/build nrf52840 UART_trans -DOT_MTD=OFF -DOT_APP_RCP=OFF -DOT_RCP=OFF
OpenThread FTD CLI বাইনারি দিয়ে ডিরেক্টরিতে নেভিগেট করুন এবং ARM এমবেডেড টুলচেনের সাহায্যে হেক্স ফরম্যাটে রূপান্তর করুন:
$ cd build/bin $ arm-none-eabi-objcopy -O ihex ot-cli-ftd ot-cli-ftd.hex
বোর্ড ফ্ল্যাশ
প্রতিটি nRF52840 বোর্ডে ot-cli-ftd.hex
ফাইলটি ফ্ল্যাশ করুন।
nRF52840 বোর্ডে বাহ্যিক পাওয়ার পিনের পাশে মাইক্রো-ইউএসবি ডিবাগ পোর্টে USB কেবলটি সংযুক্ত করুন এবং তারপরে এটি আপনার লিনাক্স মেশিনে প্লাগ করুন। সঠিকভাবে সেট করুন, LED5 চালু আছে।
আগের মতো, nRF52840 বোর্ডের সিরিয়াল নম্বরটি নোট করুন:
nRFx কমান্ড লাইন টুলের অবস্থানে নেভিগেট করুন এবং বোর্ডের সিরিয়াল নম্বর ব্যবহার করে nRF52840 বোর্ডে OpenThread CLI FTD হেক্স ফাইলটি ফ্ল্যাশ করুন:
$ cd ~/nrfjprog $ ./nrfjprog -f nrf52 -s 683704924 --verify --chiperase --program \ ~/openthread/output/nrf52840/bin/ot-cli-ftd.hex --reset
LED5 ফ্ল্যাশিংয়ের সময় সংক্ষিপ্তভাবে বন্ধ হয়ে যাবে। নিম্নলিখিত আউটপুট সাফল্যের উপর উত্পন্ন হয়:
Parsing hex file. Erasing user available code and UICR flash areas. Applying system reset. Checking that the area to write is not protected. Programing device. Applying system reset. Run.
অন্য দুটি বোর্ডের জন্য এই "বোর্ডগুলি ফ্ল্যাশ করুন" পদক্ষেপটি পুনরাবৃত্তি করুন৷ প্রতিটি বোর্ড একইভাবে লিনাক্স মেশিনের সাথে সংযুক্ত হওয়া উচিত এবং বোর্ডের ক্রমিক নম্বর ব্যতীত ফ্ল্যাশ করার কমান্ড একই। তে প্রতিটি বোর্ডের অনন্য সিরিয়াল নম্বর ব্যবহার করা নিশ্চিত করুন৷
nrfjprog
ফ্ল্যাশিং কমান্ড।
সফল হলে, প্রতিটি বোর্ডে LED1, LED2 বা LED3 আলোকিত হবে। এমনকি ফ্ল্যাশ করার পরেই আপনি 3 থেকে 2 (বা 2 থেকে 1) পর্যন্ত আলোকিত LED সুইচ দেখতে পাবেন (ডিভাইসের ভূমিকা পরিবর্তন বৈশিষ্ট্য)।
12. অ্যাপ্লিকেশন কার্যকারিতা
তিনটি nRF52840 বোর্ডই এখন চালিত হওয়া উচিত এবং আমাদের OpenThread অ্যাপ্লিকেশনটি চালানো উচিত। পূর্বে বিস্তারিত হিসাবে, এই অ্যাপ্লিকেশন দুটি প্রাথমিক বৈশিষ্ট্য আছে.
ডিভাইস ভূমিকা সূচক
প্রতিটি বোর্ডে আলোকিত LED থ্রেড নোডের বর্তমান ভূমিকা প্রতিফলিত করে:
- LED1 = নেতা
- LED2 = রাউটার
- LED3 = শেষ ডিভাইস
ভূমিকা যেমন পরিবর্তিত হয়, তেমনি আলোকিত LEDও হয়। প্রতিটি ডিভাইস পাওয়ার আপ হওয়ার 20 সেকেন্ডের মধ্যে একটি বা দুটি বোর্ডে এই পরিবর্তনগুলি আপনার ইতিমধ্যেই দেখা উচিত।
ইউডিপি মাল্টিকাস্ট
যখন একটি বোর্ডে Button1 চাপানো হয়, তখন একটি UDP বার্তা মেশ-স্থানীয় মাল্টিকাস্ট ঠিকানায় পাঠানো হয়, যা থ্রেড নেটওয়ার্কের অন্যান্য সমস্ত নোড অন্তর্ভুক্ত করে। এই বার্তাটি পাওয়ার প্রতিক্রিয়া হিসাবে, অন্যান্য সমস্ত বোর্ডে LED4 চালু বা বন্ধ টগল করে । প্রতিটি বোর্ডের জন্য LED4 চালু বা বন্ধ থাকে যতক্ষণ না এটি অন্য UDP বার্তা না পায়।
13. ডেমো: ডিভাইসের ভূমিকা পরিবর্তন পর্যবেক্ষণ করুন
আপনি যে ডিভাইসগুলি ফ্ল্যাশ করেছেন তা হল একটি নির্দিষ্ট ধরণের ফুল থ্রেড ডিভাইস (এফটিডি) যাকে রাউটার এলিজিবল এন্ড ডিভাইস (REED) বলা হয়। এর অর্থ হল তারা একটি রাউটার বা এন্ড ডিভাইস হিসাবে কাজ করতে পারে এবং একটি এন্ড ডিভাইস থেকে রাউটারে নিজেদের প্রচার করতে পারে।
থ্রেড 32টি রাউটার পর্যন্ত সমর্থন করতে পারে, কিন্তু রাউটারের সংখ্যা 16 থেকে 23 এর মধ্যে রাখার চেষ্টা করে। যদি একটি REED একটি এন্ড ডিভাইস হিসাবে সংযুক্ত করে এবং রাউটারের সংখ্যা 16-এর নিচে হয়, তাহলে এটি স্বয়ংক্রিয়ভাবে একটি রাউটারে নিজেকে উন্নীত করে। আপনি অ্যাপ্লিকেশনে (20 সেকেন্ড) otThreadSetRouterSelectionJitter
মান সেট করেছেন এমন সেকেন্ডের মধ্যে এই পরিবর্তনটি র্যান্ডম সময়ে হওয়া উচিত।
প্রতিটি থ্রেড নেটওয়ার্কেরও একটি লিডার থাকে, যা একটি রাউটার যা একটি থ্রেড নেটওয়ার্কে রাউটারের সেট পরিচালনার জন্য দায়ী। সমস্ত ডিভাইস চালু থাকলে, 20 সেকেন্ড পরে তাদের মধ্যে একটি লিডার (LED1 চালু) এবং অন্য দুটি রাউটার (LED2 চালু) হওয়া উচিত।
নেতাকে সরান
যদি থ্রেড নেটওয়ার্ক থেকে লিডারকে সরিয়ে দেওয়া হয়, তাহলে একটি ভিন্ন রাউটার নিজেকে একজন লিডারে উন্নীত করে, যাতে নিশ্চিত করা যায় যে নেটওয়ার্কটিতে এখনও একজন লিডার আছে।
পাওয়ার সুইচ ব্যবহার করে লিডার বোর্ড (যেটিতে LED1 আলো আছে) বন্ধ করুন। প্রায় 20 সেকেন্ড অপেক্ষা করুন। বাকি দুটি বোর্ডের একটিতে, LED2 (রাউটার) বন্ধ হয়ে যাবে এবং LED1 (লিডার) চালু হবে। এই ডিভাইসটি এখন থ্রেড নেটওয়ার্কের নেতা।
আসল লিডার বোর্ড আবার চালু করুন। এটি স্বয়ংক্রিয়ভাবে থ্রেড নেটওয়ার্কে একটি শেষ ডিভাইস হিসাবে পুনরায় যোগদান করা উচিত (LED3 জ্বলছে)। 20 সেকেন্ডের মধ্যে (রাউটার নির্বাচন জিটার) এটি নিজেকে একটি রাউটারে উন্নীত করে (LED2 আলোকিত)।
বোর্ডগুলি পুনরায় সেট করুন
তিনটি বোর্ডই বন্ধ করুন, তারপরে তাদের আবার চালু করুন এবং LED গুলি পর্যবেক্ষণ করুন। চালিত প্রথম বোর্ডটি লিডার রোলে শুরু হওয়া উচিত (LED1 জ্বলছে)—থ্রেড নেটওয়ার্কের প্রথম রাউটারটি স্বয়ংক্রিয়ভাবে লিডার হয়ে যায়।
অন্য দুটি বোর্ড প্রাথমিকভাবে এন্ড ডিভাইস (LED3 আলোকিত) হিসাবে নেটওয়ার্কের সাথে সংযুক্ত হয় কিন্তু 20 সেকেন্ডের মধ্যে নিজেদের রাউটারে (LED2 আলোকিত) প্রচার করা উচিত।
নেটওয়ার্ক পার্টিশন
যদি আপনার বোর্ডগুলি পর্যাপ্ত শক্তি না পায়, বা তাদের মধ্যে রেডিও সংযোগ দুর্বল হয়, তাহলে থ্রেড নেটওয়ার্কটি পার্টিশনে বিভক্ত হতে পারে এবং আপনার কাছে একাধিক ডিভাইস থাকতে পারে যা লিডার হিসাবে প্রদর্শিত হতে পারে।
থ্রেড স্ব-নিরাময় হয়, তাই পার্টিশনগুলি শেষ পর্যন্ত একটি নেতার সাথে একটি একক পার্টিশনে আবার একত্রিত হওয়া উচিত।
14. ডেমো: UDP মাল্টিকাস্ট পাঠান
পূর্ববর্তী অনুশীলন থেকে অব্যাহত থাকলে, LED4 কোনো ডিভাইসে আলো দেওয়া উচিত নয়।
যেকোনো বোর্ড বেছে নিন এবং বোতাম 1 টিপুন। অ্যাপ্লিকেশানটি চলমান থ্রেড নেটওয়ার্কের অন্যান্য সমস্ত বোর্ডে LED4 তাদের অবস্থা টগল করা উচিত। আগের ব্যায়াম থেকে অব্যাহত থাকলে, সেগুলি এখন চালু হওয়া উচিত।
একই বোর্ডের জন্য আবার বোতাম 1 টিপুন। অন্য সব বোর্ডে LED4 আবার টগল করা উচিত।
একটি ভিন্ন বোর্ডে বোতাম 1 টিপুন এবং অন্যান্য বোর্ডে কীভাবে LED4 টগল হয় তা পর্যবেক্ষণ করুন। LED4 বর্তমানে চালু আছে এমন একটি বোর্ডে বোতাম 1 টিপুন। LED4 সেই বোর্ডের জন্য চালু থাকে কিন্তু অন্যগুলিতে টগল করে।
নেটওয়ার্ক পার্টিশন
যদি আপনার বোর্ডগুলি বিভক্ত হয়ে থাকে এবং তাদের মধ্যে একাধিক নেতা থাকে, তাহলে মাল্টিকাস্ট বার্তার ফলাফল বোর্ডগুলির মধ্যে আলাদা হবে৷ আপনি যদি পার্টিশন করা একটি বোর্ডে বোতাম 1 টিপুন (এবং এইভাবে বিভাজিত থ্রেড নেটওয়ার্কের একমাত্র সদস্য), অন্য বোর্ডগুলিতে LED4 প্রতিক্রিয়াতে আলোকিত হবে না। যদি এটি ঘটে, বোর্ডগুলি পুনরায় সেট করুন - আদর্শভাবে তারা একটি একক থ্রেড নেটওয়ার্ক সংস্কার করবে এবং UDP মেসেজিং সঠিকভাবে কাজ করবে৷
15. অভিনন্দন!
আপনি একটি অ্যাপ্লিকেশন তৈরি করেছেন যা OpenThread API ব্যবহার করে!
আপনি এখন জানেন:
- নর্ডিক nRF52840 ডেভ বোর্ডে বোতাম এবং এলইডি কীভাবে প্রোগ্রাম করবেন
- কিভাবে সাধারণ OpenThread API এবং
otInstance
ক্লাস ব্যবহার করবেন - কিভাবে ওপেন থ্রেড অবস্থার পরিবর্তনগুলি নিরীক্ষণ এবং প্রতিক্রিয়া জানাতে হয়
- কিভাবে একটি থ্রেড নেটওয়ার্কের সমস্ত ডিভাইসে UDP বার্তা পাঠাতে হয়
- মেকফাইলগুলি কীভাবে পরিবর্তন করবেন
পরবর্তী পদক্ষেপ
এই কোডল্যাবটি তৈরি করে, নিম্নলিখিত অনুশীলনগুলি চেষ্টা করুন:
- অনবোর্ড এলইডির পরিবর্তে জিপিআইও পিন ব্যবহার করতে জিপিআইও মডিউলটি পরিবর্তন করুন এবং রাউটারের ভূমিকার উপর ভিত্তি করে রঙ পরিবর্তন করে বহিরাগত আরজিবি এলইডি সংযুক্ত করুন
- একটি ভিন্ন উদাহরণ প্ল্যাটফর্মের জন্য GPIO সমর্থন যোগ করুন
- একটি বোতাম প্রেস থেকে সমস্ত ডিভাইস পিং করার জন্য মাল্টিকাস্ট ব্যবহার করার পরিবর্তে, একটি পৃথক ডিভাইস সনাক্ত করতে এবং পিং করতে রাউটার/লিডার API ব্যবহার করুন
- একটি ওপেন থ্রেড বর্ডার রাউটার ব্যবহার করে আপনার মেশ নেটওয়ার্ককে ইন্টারনেটে সংযুক্ত করুন এবং LED গুলিকে আলোকিত করতে থ্রেড নেটওয়ার্কের বাইরে থেকে মাল্টিকাস্ট করুন
আরও পড়া
Openthread.io এবং GitHub বিভিন্ন OpenThread সম্পদের জন্য দেখুন, যার মধ্যে রয়েছে:
- সমর্থিত প্ল্যাটফর্ম - OpenThread সমর্থন করে এমন সমস্ত প্ল্যাটফর্ম আবিষ্কার করুন
- OpenThread তৈরি করুন — OpenThread নির্মাণ এবং কনফিগার করার আরও বিশদ বিবরণ
- থ্রেড প্রাইমার - থ্রেড ধারণার একটি দুর্দান্ত রেফারেন্স
তথ্যসূত্র: