HIDL ডেটা ঘোষণা C++ স্ট্যান্ডার্ড-লেআউট ডেটা স্ট্রাকচার তৈরি করে। এই স্ট্রাকচারগুলি যে কোনও জায়গায় স্থাপন করা যেতে পারে যা প্রাকৃতিক মনে হয় (স্ট্যাকের উপর, ফাইলে বা গ্লোবাল স্কোপে, বা স্তূপে) এবং একই ফ্যাশনে রচনা করা যেতে পারে। ক্লায়েন্ট কোড এইচআইডিএল প্রক্সি কোডকে কনস্ট রেফারেন্স এবং আদিম প্রকারে পাসিং বলে, যখন স্টাব এবং প্রক্সি কোড সিরিয়ালাইজেশনের বিবরণ লুকিয়ে রাখে।
দ্রষ্টব্য: ডেটা স্ট্রাকচারকে স্পষ্টভাবে সিরিয়ালাইজ বা ডিসিরিয়ালাইজ করার জন্য ডেভেলপার-লিখিত কোডের প্রয়োজন নেই।
নীচের সারণীটি HIDL আদিমকে C++ ডেটা প্রকারের সাথে মানচিত্র করে:
HIDL প্রকার | C++ প্রকার | হেডার/লাইব্রেরি |
---|---|---|
enum | enum class | |
uint8_t..uint64_t | uint8_t..uint64_t | <stdint.h> |
int8_t..int64_t | int8_t..int64_t | <stdint.h> |
float | float | |
double | double | |
vec<T> | hidl_vec<T> | libhidlbase |
T[S1][S2]...[SN] | T[S1][S2]...[SN] | |
string | hidl_string | libhidlbase |
handle | hidl_handle | libhidlbase |
safe_union | (custom) struct | |
struct | struct | |
union | union | |
fmq_sync | MQDescriptorSync | libhidlbase |
fmq_unsync | MQDescriptorUnsync | libhidlbase |
নীচের বিভাগগুলি আরও বিস্তারিতভাবে ডেটা প্রকারগুলি বর্ণনা করে৷
enum
HIDL-এর একটি enum C++ এ একটি enum হয়। যেমন:
enum Mode : uint8_t { WRITE = 1 << 0, READ = 1 << 1 }; enum SpecialMode : Mode { NONE = 0, COMPARE = 1 << 2 };
… হয়ে যায়:
enum class Mode : uint8_t { WRITE = 1, READ = 2 }; enum class SpecialMode : uint8_t { WRITE = 1, READ = 2, NONE = 0, COMPARE = 4 };
Android 10 থেকে শুরু করে, ::android::hardware::hidl_enum_range
ব্যবহার করে একটি enum পুনরাবৃত্তি করা যেতে পারে। এই পরিসরটি HIDL সোর্স কোডে প্রদর্শিত ক্রমানুসারে প্রতিটি গণনাকারীকে অন্তর্ভুক্ত করে, পিতামাতার এনাম থেকে শুরু করে শেষ সন্তান পর্যন্ত। উদাহরণস্বরূপ, এই কোডটি সেই ক্রমে WRITE
, READ
, NONE
, এবং COMPARE
এর উপর পুনরাবৃত্তি করে৷ উপরে দেওয়া SpecialMode
:
template <typename T> using hidl_enum_range = ::android::hardware::hidl_enum_range<T> for (SpecialMode mode : hidl_enum_range<SpecialMode>) {...}
hidl_enum_range
এছাড়াও বিপরীত পুনরাবৃত্তিকারী প্রয়োগ করে এবং constexpr
প্রসঙ্গে ব্যবহার করা যেতে পারে। যদি একটি মান একাধিকবার একটি গণনায় প্রদর্শিত হয়, মানটি একাধিকবার পরিসরে প্রদর্শিত হবে।
বিটফিল্ড<T>
bitfield<T>
(যেখানে T
একটি ব্যবহারকারী-সংজ্ঞায়িত enum) C++-এ সেই enum-এর অন্তর্নিহিত ধরণ হয়ে ওঠে। উপরের উদাহরণে, bitfield<Mode>
হয়ে যায় uint8_t
।
vec<T>
hidl_vec<T>
ক্লাস টেমপ্লেটটি libhidlbase
এর অংশ এবং এটিকে ইচ্ছামত আকারের সাথে যেকোনো HIDL প্রকারের ভেক্টর পাস করতে ব্যবহার করা যেতে পারে। তুলনাযোগ্য নির্দিষ্ট আকারের ধারক হল hidl_array
। hidl_vec::setToExternal()
ফাংশন ব্যবহার করে T
টাইপের একটি বাহ্যিক ডেটা বাফার নির্দেশ করার জন্য একটি hidl_vec<T>
ও শুরু করা যেতে পারে।
জেনারেট করা C++ হেডারে যথাযথভাবে স্ট্রাকট নির্গত/ঢোকানোর পাশাপাশি, vec<T>
ব্যবহার std::vector
এবং বেয়ার T
পয়েন্টার থেকে অনুবাদ করার জন্য কিছু সুবিধাজনক ফাংশন তৈরি করে। যদি vec<T>
একটি প্যারামিটার হিসাবে ব্যবহার করা হয়, তাহলে যে ফাংশনটি ব্যবহার করে সেটি ওভারলোড হয় (দুটি প্রোটোটাইপ তৈরি হয়) সেই প্যারামিটারের জন্য HIDL স্ট্রাকট এবং একটি std::vector<T>
টাইপ উভয়ই গ্রহণ এবং পাস করতে।
অ্যারে
hidl-এ ধ্রুবক অ্যারে libhidlbase
এ hidl_array
ক্লাস দ্বারা উপস্থাপিত হয়। একটি hidl_array<T, S1, S2, …, SN>
একটি N ডাইমেনশনাল ফিক্সড সাইজ অ্যারে T[S1][S2]…[SN]
প্রতিনিধিত্ব করে।
স্ট্রিং
hidl_string
ক্লাস ( libhidlbase
এর অংশ) HIDL ইন্টারফেসের উপর স্ট্রিং পাস করতে ব্যবহার করা যেতে পারে এবং /system/libhidl/base/include/hidl/HidlSupport.h
এ সংজ্ঞায়িত করা হয়েছে। ক্লাসের প্রথম স্টোরেজ অবস্থানটি এর অক্ষর বাফারের একটি নির্দেশক।
hidl_string
জানে কিভাবে std::string and char*
(C-স্টাইল স্ট্রিং) থেকে operator=
, অন্তর্নিহিত কাস্ট এবং .c_str()
ফাংশন ব্যবহার করে রূপান্তর করতে হয়। HIDL স্ট্রিং স্ট্রাকটে উপযুক্ত কপি কনস্ট্রাক্টর এবং অ্যাসাইনমেন্ট অপারেটর রয়েছে:
- একটি
std::string
বা একটি C স্ট্রিং থেকে HIDL স্ট্রিং লোড করুন। - একটি HIDL স্ট্রিং থেকে একটি নতুন
std::string
তৈরি করুন।
উপরন্তু, HIDL স্ট্রিংগুলিতে রূপান্তর কনস্ট্রাক্টর রয়েছে তাই C স্ট্রিং ( char *
) এবং C++ স্ট্রিংগুলি ( std::string
) একটি HIDL স্ট্রিং নেওয়া পদ্ধতিতে ব্যবহার করা যেতে পারে।
গঠন
HIDL-এর একটি struct
শুধুমাত্র নির্দিষ্ট-আকারের ডেটা প্রকার থাকতে পারে এবং কোনও ফাংশন নেই। HIDL struct সংজ্ঞাগুলি সরাসরি C++-এ স্ট্যান্ডার্ড-লেআউট struct
s-এ ম্যাপ করে, নিশ্চিত করে যে struct
এর একটি সামঞ্জস্যপূর্ণ মেমরি বিন্যাস রয়েছে। একটি স্ট্রাকটে HIDL প্রকারগুলি অন্তর্ভুক্ত থাকতে পারে, handle
, string
, এবং vec<T>
সহ, যা পৃথক পরিবর্তনশীল-দৈর্ঘ্য বাফারের দিকে নির্দেশ করে।
হ্যান্ডেল
সতর্কতা: যেকোনো ধরনের ঠিকানা (এমনকি শারীরিক ডিভাইসের ঠিকানা) কখনই নেটিভ হ্যান্ডেলের অংশ হতে হবে না। প্রক্রিয়াগুলির মধ্যে এই তথ্যটি পাস করা বিপজ্জনক এবং তাদের আক্রমণের জন্য সংবেদনশীল করে তোলে। একটি প্রক্রিয়ার মধ্যে বরাদ্দ করা মেমরি খোঁজার জন্য ব্যবহার করার আগে প্রক্রিয়াগুলির মধ্যে পাস করা যেকোনো মান অবশ্যই যাচাই করা উচিত। অন্যথায়, খারাপ হ্যান্ডেলগুলি খারাপ মেমরি অ্যাক্সেস বা মেমরি দুর্নীতির কারণ হতে পারে।
handle
ধরনটি C++-এ hidl_handle
কাঠামো দ্বারা উপস্থাপিত হয়, যা একটি const native_handle_t
অবজেক্টের পয়েন্টারের চারপাশে একটি সাধারণ মোড়ক (এটি অনেক দিন ধরে অ্যান্ড্রয়েডে উপস্থিত রয়েছে)।
typedef struct native_handle { int version; /* sizeof(native_handle_t) */ int numFds; /* number of file descriptors at &data[0] */ int numInts; /* number of ints at &data[numFds] */ int data[0]; /* numFds + numInts ints */ } native_handle_t;
ডিফল্টরূপে, hidl_handle
native_handle_t
পয়েন্টারের মালিকানা নেয় না যা এটি মোড়ানো হয়। এটি শুধুমাত্র একটি native_handle_t
তে একটি পয়েন্টারকে নিরাপদে সংরক্ষণ করার জন্য বিদ্যমান যাতে এটি 32- এবং 64-বিট উভয় প্রক্রিয়াতেই ব্যবহার করা যেতে পারে।
যেসব পরিস্থিতিতে hidl_handle
এর আবদ্ধ ফাইল বর্ণনাকারীর মালিকানা রয়েছে তার মধ্যে রয়েছে:
-
setTo(native_handle_t* handle, bool shouldOwn)
পদ্ধতিতে একটি কল অনুসরণ করেshouldOwn
প্যারামিটারটিtrue
সেট করা হয়েছে - যখন
hidl_handle
অবজেক্ট অন্যhidl_handle
অবজেক্ট থেকে কপি নির্মাণের মাধ্যমে তৈরি করা হয় - যখন
hidl_handle
অবজেক্ট অন্যhidl_handle
অবজেক্ট থেকে কপি-অ্যাসাইন করা হয়
hidl_handle
native_handle_t*
অবজেক্টে/থেকে অন্তর্নিহিত এবং স্পষ্ট উভয় রূপান্তর প্রদান করে। HIDL-এ handle
টাইপের প্রধান ব্যবহার হল HIDL ইন্টারফেসের উপর ফাইল বর্ণনাকারী পাস করা। তাই একটি একক ফাইল বর্ণনাকারীকে int
s এবং একটি একক fd
ছাড়াই একটি native_handle_t
দ্বারা উপস্থাপন করা হয়। যদি ক্লায়েন্ট এবং সার্ভার একটি ভিন্ন প্রক্রিয়ায় থাকে, RPC বাস্তবায়ন স্বয়ংক্রিয়ভাবে ফাইল বর্ণনাকারীর যত্ন নেয় যাতে উভয় প্রক্রিয়া একই ফাইলে কাজ করতে পারে।
যদিও একটি প্রক্রিয়ার দ্বারা একটি hidl_handle
এ প্রাপ্ত একটি ফাইল বর্ণনাকারী সেই প্রক্রিয়ায় বৈধ, তবে এটি প্রাপ্তির ফাংশনের বাইরে টিকে থাকে না (ফাংশনটি ফিরে আসলে এটি বন্ধ হয়ে যায়)। একটি প্রক্রিয়া যা ফাইল বর্ণনাকারীতে অবিরাম অ্যাক্সেস বজায় রাখতে চায় তাকে অবশ্যই আবদ্ধ ফাইল বর্ণনাকারীকে dup()
হবে, অথবা সম্পূর্ণ hidl_handle
অবজেক্টটি অনুলিপি করতে হবে।
স্মৃতি
HIDL memory
টাইপ ম্যাপ libhidlbase
এ hidl_memory
ক্লাসে, যা আনম্যাপ করা শেয়ার করা মেমরির প্রতিনিধিত্ব করে। এটি এমন বস্তু যা HIDL-এ মেমরি ভাগ করার জন্য প্রক্রিয়াগুলির মধ্যে পাস করতে হবে। শেয়ার করা মেমরি ব্যবহার করতে:
-
IAllocator
এর একটি উদাহরণ প্রাপ্ত করুন (বর্তমানে শুধুমাত্র উদাহরণ "ashmem" উপলব্ধ) এবং শেয়ার করা মেমরি বরাদ্দ করতে এটি ব্যবহার করুন। -
IAllocator::allocate()
একটিhidl_memory
অবজেক্ট প্রদান করে যা HIDL RPC এর মাধ্যমে পাস করা যেতে পারে এবংlibhidlmemory
এরmapMemory
ফাংশন ব্যবহার করে একটি প্রক্রিয়াতে ম্যাপ করা যেতে পারে। -
mapMemory
একটিsp<IMemory>
বস্তুর একটি রেফারেন্স প্রদান করে যা মেমরি অ্যাক্সেস করতে ব্যবহার করা যেতে পারে। (IMemory
এবংIAllocator
android.hidl.memory@1.0
এ সংজ্ঞায়িত করা হয়েছে।)
IAllocator
এর একটি উদাহরণ মেমরি বরাদ্দ করতে ব্যবহার করা যেতে পারে:
#include <android/hidl/allocator/1.0/IAllocator.h> #include <android/hidl/memory/1.0/IMemory.h> #include <hidlmemory/mapping.h> using ::android::hidl::allocator::V1_0::IAllocator; using ::android::hidl::memory::V1_0::IMemory; using ::android::hardware::hidl_memory; .... sp<IAllocator> ashmemAllocator = IAllocator::getService("ashmem"); ashmemAllocator->allocate(2048, [&](bool success, const hidl_memory& mem) { if (!success) { /* error */ } // now you can use the hidl_memory object 'mem' or pass it around }));
মেমরির প্রকৃত পরিবর্তন অবশ্যই একটি IMemory
অবজেক্টের মাধ্যমে করা উচিত, যেটি mem
তৈরি করেছে বা যে পাশে এটি HIDL RPC এর মাধ্যমে গ্রহণ করে।
// Same includes as above sp<IMemory> memory = mapMemory(mem); void* data = memory->getPointer(); memory->update(); // update memory however you wish after calling update and before calling commit data[0] = 42; memory->commit(); // … memory->update(); // the same memory can be updated multiple times // … memory->commit();
ইন্টারফেস
ইন্টারফেস বস্তু হিসাবে পাস করা যেতে পারে. android.hidl.base@1.0::IBase
টাইপের জন্য ইন্টারফেস শব্দটি সিনট্যাকটিক চিনি হিসাবে ব্যবহার করা যেতে পারে; উপরন্তু, বর্তমান ইন্টারফেস এবং কোনো আমদানি করা ইন্টারফেস একটি প্রকার হিসাবে সংজ্ঞায়িত করা হয়।
ইন্টারফেস ধারণকারী ভেরিয়েবলগুলি শক্তিশালী পয়েন্টার হওয়া উচিত: sp<IName>
। এইচআইডিএল ফাংশন যা ইন্টারফেস প্যারামিটার গ্রহণ করে সেগুলি কাঁচা পয়েন্টারকে শক্তিশালী পয়েন্টারে রূপান্তর করে, যার ফলে অপ্রত্যাশিত আচরণ হয় (পয়েন্টারটি অপ্রত্যাশিতভাবে পরিষ্কার করা যেতে পারে)। সমস্যা এড়াতে, সবসময় HIDL ইন্টারফেস একটি sp<>
হিসাবে সংরক্ষণ করুন।