Android 10 के साथ बेहतर तरीके से काम करने वाला Android इंटरफ़ेस डेफ़िनिशन लैंग्वेज (एआईडीएल), ऐप्लिकेशन प्रोग्राम को ट्रैक करने का एक नया तरीका है AIDL से मिला इंटरफ़ेस (एपीआई) और ऐप्लिकेशन बाइनरी इंटरफ़ेस (एबीआई) इंटरफ़ेस. स्थिर एआईडीएल बिलकुल एआईडीएल की तरह काम करता है, लेकिन बिल्ड सिस्टम ट्रैक करता है इंटरफ़ेस काम करता है और इन पर कुछ पाबंदियां लगाई जाती हैं:
- इंटरफ़ेस को बिल्ड सिस्टम में
aidl_interfaces
के साथ तय किया जाता है. - इंटरफ़ेस में सिर्फ़ स्ट्रक्चर्ड डेटा हो सकता है. पार्स किए जा सकने वाले ऐसे एलिमेंट जो एआईडीएल की परिभाषा के आधार पर, पसंदीदा टाइप अपने-आप बन जाते हैं और अपने-आप मार्शल और अनमार्शल हो जाते हैं.
- इंटरफ़ेस को स्टेबल (पुराने वर्शन के साथ काम करने वाले) के तौर पर बताया जा सकता है. जब यह ऐसा होता है, तो उसके एपीआई को ट्रैक किया जाता है और एआईडीएल के बगल में मौजूद फ़ाइल में उसका वर्शन बनाया जाता है इंटरफ़ेस पर कॉपी करने की सुविधा मिलती है.
स्ट्रक्चर्ड बनाम स्टेबल एआईडीएल
स्ट्रक्चर्ड एआईडीएल का मतलब ऐसे टाइप से है जिन्हें पूरी तरह से एआईडीएल में बताया गया है. उदाहरण के लिए, पार्स किए जा सकने वाले डिक्लेरेशन (पसंद के मुताबिक पार्स किया जा सकने वाला) का फ़ॉर्मैट एआईडीएल नहीं है. पार्स किए जा सकने वाले एआईडीएल में तय किए गए फ़ील्ड को स्ट्रक्चर्ड पार्सलेबल कहा जाता है.
स्टेबल एआईडीएल को स्ट्रक्चर्ड एआईडीएल की ज़रूरत होती है, ताकि बिल्ड सिस्टम और कंपाइलर
यह समझ सकते हैं कि पार्स किए जा सकने वाले विज्ञापनों में किए गए बदलाव, पुराने सिस्टम के साथ काम करते हैं या नहीं.
हालांकि, सभी स्ट्रक्चर्ड इंटरफ़ेस स्टेबल नहीं होते. स्थिर रहने के लिए,
इंटरफ़ेस में सिर्फ़ स्ट्रक्चर्ड टाइप का इस्तेमाल होना चाहिए. साथ ही, इसमें नीचे दी गई चीज़ों का भी इस्तेमाल होना चाहिए
वर्शन बनाने की सुविधा देता है. इसके उलट, मुख्य बिल्ड के दौरान इंटरफ़ेस स्थिर नहीं होता
सिस्टम का इस्तेमाल उसे बनाने के लिए किया जाता है या unstable:true
सेट है.
एआईडीएल इंटरफ़ेस तय करना
aidl_interface
की परिभाषा इस तरह दिखती है:
aidl_interface {
name: "my-aidl",
srcs: ["srcs/aidl/**/*.aidl"],
local_include_dir: "srcs/aidl",
imports: ["other-aidl"],
versions_with_info: [
{
version: "1",
imports: ["other-aidl-V1"],
},
{
version: "2",
imports: ["other-aidl-V3"],
}
],
stability: "vintf",
backend: {
java: {
enabled: true,
platform_apis: true,
},
cpp: {
enabled: true,
},
ndk: {
enabled: true,
},
rust: {
enabled: true,
},
},
}
name
: एआईडीएल इंटरफ़ेस मॉड्यूल का नाम, जो किसी खास तरह के एआईडीएल इंटरफ़ेस एआईडीएल इंटरफ़ेस.srcs
: इंटरफ़ेस बनाने वाली AIDL सोर्स फ़ाइलों की सूची. रास्ताcom.acme
पैकेज में तय किए गए AIDL टाइपFoo
के लिए यह होना चाहिए:<base_path>/com/acme/Foo.aidl
, जहां<base_path>
कोई भी डायरेक्ट्री हो सकती है उस डायरेक्ट्री से जुड़ी होगी जिसमेंAndroid.bp
मौजूद है. पिछले उदाहरण में,<base_path>
,srcs/aidl
है.local_include_dir
: वह पाथ जहां से पैकेज का नाम शुरू होता है. यह ऊपर बताई गई<base_path>
से मेल खाती है.imports
:aidl_interface
मॉड्यूल की सूची जिनका यह इस्तेमाल किया जाता है. अगर आपके एआईडीएल इंटरफ़ेस, ऐसे इंटरफ़ेस या किसी दूसरे इंटरफ़ेस के पार्स किए जा सकने वाले इंटरफ़ेस का इस्तेमाल करते हैंaidl_interface
, उसका नाम यहां रखें. अपने-आप में यह नाम हो सकता है, ताकि आपको सबसे नई वर्शन या वर्शन के सफ़िक्स के साथ नाम (जैसे कि-V1
), जिसे की है. यह तय करना कि Android 12 और इसके बाद के वर्शन का इस्तेमाल किया जा रहा है या नहींversions
: इंटरफ़ेस के पिछले वर्शन जोapi_dir
में फ़्रीज़ किए गए वर्शन. Android 11 औरversions
,aidl_api/name
में फ़्रीज़ किए गए हैं. अगर किसी इंटरफ़ेस का कोई फ़्रीज़ किया गया वर्शन नहीं है, इसे नहीं बताया जाना चाहिए और इसके साथ काम करने की जांच नहीं की जाएगी. Android के लिए, इस फ़ील्ड कोversions_with_info
से बदल दिया गया है 13 और उससे ज़्यादा.versions_with_info
: टपल की सूची, जिनमें से हर एक में फ़्रीज़ किया गया वर्शन और किसी अन्य aidl_interface के वर्शन को इंपोर्ट करने वाली सूची ऐसे मॉड्यूल जिन्हें aidl_interface के इस वर्शन से इंपोर्ट किया गया था. परिभाषा एक AIDL इंटरफ़ेस IFACE के वर्शन Vaidl_api/IFACE/V
. इस फ़ील्ड को Android 13 में पेश किया गया था. और इसे सीधेAndroid.bp
में बदला नहीं जाना चाहिए. फ़ील्ड है*-update-api
या*-freeze-api
का इस्तेमाल करके, जोड़ा या अपडेट किया गया. साथ ही,versions
फ़ील्डversions_with_info
पर अपने-आप माइग्रेट हो जाते हैं जब कोई उपयोगकर्ता*-update-api
या*-freeze-api
को शुरू करता है.stability
: इस इंटरफ़ेस की स्थिरता का वादा करने के लिए वैकल्पिक फ़्लैग. यह सिर्फ़"vintf"
के साथ काम करता है. अगरstability
सेट नहीं है, तो बिल्ड सिस्टम यह जांच करता है कि इंटरफ़ेस पुराने सिस्टम के साथ काम करता है या नहींunstable
बताया गया है. अगर नीति को सेट नहीं किया जाता है, तो यह स्थिरता के लिए किया जा सकता है (इसलिए, सिस्टम की सभी चीज़ों को, उदाहरण के लिए,system.img
और उससे जुड़े पार्टिशन या सभी वेंडर में मौजूद चीज़ें चीज़ें, जैसे किvendor.img
और उससे जुड़े पार्टिशन). अगर आपनेstability
को"vintf"
पर सेट किया गया है. यह स्थिरता का वादा करता है: जब तक इंटरफ़ेस उपयोग किया जाता है, तब तक उसे स्थिर रखना चाहिए.gen_trace
: ट्रेस करने की सुविधा को चालू या बंद करने के लिए वैकल्पिक फ़्लैग. इतने समय में शुरू होगा Android 14,cpp
के लिए डिफ़ॉल्ट तौर परtrue
है.java
बैकएंड.host_supported
: वैकल्पिक फ़्लैग, जिसेtrue
पर सेट करने से जनरेट की गई लाइब्रेरी, जो होस्ट एनवायरमेंट में उपलब्ध हैं.unstable
: वैकल्पिक फ़्लैग का इस्तेमाल यह बताने के लिए किया जाता है कि यह इंटरफ़ेस काम नहीं करता स्थिर होना चाहिए. जब इसेtrue
पर सेट किया जाता है, तो बिल्ड सिस्टम में से दोनों में से कोई भी नहीं सेट होता इंटरफ़ेस के लिए एपीआई डंप बनाता है और न ही उसे अपडेट करने की ज़रूरत होती है.frozen
: वैकल्पिक फ़्लैग कोtrue
पर सेट करने का मतलब है कि इंटरफ़ेस इंटरफ़ेस के पिछले वर्शन के बाद से कोई बदलाव नहीं हुआ है. यह चालू करता है सुरक्षा की जांच करनी होती है. अगर इसेfalse
पर सेट किया जाता है, तो इसका मतलब है कि इंटरफ़ेस जिसमें नए बदलाव किए गए हैं, इसलिएfoo-freeze-api
को चलाने से एक नया वर्शन डाउनलोड करने के लिए लिंक किया गया है और वैल्यू को अपने-आपtrue
में बदल दिया जाएगा. पहली बार इसमें दिखाया गया Android 14.backend.<type>.enabled
: ये फ़्लैग, हर उस बैकएंड को टॉगल करते हैं तो AIDL कंपाइलर इसके लिए कोड जनरेट करता है. चार बैकएंड हैं समर्थित: Java, C++, NDK, और Rust. Java, C++, और NDK बैकएंड चालू हैं डिफ़ॉल्ट रूप से. अगर इन तीनों में से किसी भी बैकएंड की ज़रूरत नहीं है, तो उसे अक्षम कर दिया जाएगा. Android तक, Rust की सुविधा डिफ़ॉल्ट रूप से बंद रहेगी 15 (एओएसपी एक्सपेरिमेंट के तौर पर उपलब्ध है).backend.<type>.apex_available
: जनरेट किए गए APEX नामों की सूची स्टब लाइब्रेरी यहां उपलब्ध है.backend.[cpp|java].gen_log
: यह एक वैकल्पिक फ़्लैग है, जो तय करता है कि लेन-देन की जानकारी इकट्ठा करने के लिए, अतिरिक्त कोड जनरेट करना.backend.[cpp|java].vndk.enabled
: यह इंटरफ़ेस बनाने के लिए वैकल्पिक फ़्लैग का हिस्सा है. डिफ़ॉल्ट वैल्यूfalse
है.backend.[cpp|ndk].additional_shared_libraries
: पहली बार इसमें दिखाया गया यह फ़्लैग, Android 14 के लिए डिपेंडेंसी स्थानीय लाइब्रेरी. यह फ़्लैगndk_header
औरcpp_header
के लिए काम का है.backend.java.sdk_version
: वर्शन तय करने के लिए वैकल्पिक फ़्लैग के हिसाब से बनाया गया है जिसके लिए Java स्टब लाइब्रेरी बनाई गई है. डिफ़ॉल्ट सेटिंग यह है"system_current"
. अगरbackend.java.platform_apis
है, तो इसे सेट नहीं करना चाहिएtrue
है.backend.java.platform_apis
: वह वैकल्पिक फ़्लैग जिसे इस पर सेट किया जाना चाहिएtrue
जब जनरेट की गई लाइब्रेरी को प्लैटफ़ॉर्म एपीआई के ज़रिए बनाना हो इस्तेमाल करें.
वर्शन के हर कॉम्बिनेशन और चालू बैकएंड के लिए, एक स्टब लाइब्रेरी बनाई जाती है. स्टब लाइब्रेरी के खास वर्शन के बारे में बताने के लिए किसी खास बैकएंड के लिए, मॉड्यूल का नाम रखने के नियम देखें.
AIDL फ़ाइलें लिखें
स्थिर एआईडीएल में मौजूद इंटरफ़ेस, पारंपरिक इंटरफ़ेस की तरह ही होते हैं. इसके अपवाद हैं, क्योंकि उन्हें अव्यवस्थित पार्स किए जा सकने वाले दस्तावेज़ों का इस्तेमाल करने की अनुमति नहीं है (क्योंकि ये स्थिर नहीं हैं! स्ट्रक्चर्ड बनाम स्टेबल एआईडीएल). स्थिर एआईडीएल में मुख्य अंतर यह है कि पार्स किए जा सकते हैं. पहले, पार्सल किए जा सकने वाले प्रॉडक्ट का एलान किया जाता था; इंच एआईडीएल, पार्स किए जा सकने वाले फ़ील्ड, और वैरिएबल परिभाषित किया जा सकता है.
// in a file like 'some/package/Thing.aidl'
package some.package;
parcelable SubThing {
String a = "foo";
int b;
}
boolean
, char
, के लिए डिफ़ॉल्ट तौर पर इस्तेमाल किया जा सकता है. हालांकि, यह ज़रूरी नहीं है,
float
, double
, byte
, int
, long
, और String
. Android में
12, उपयोगकर्ता-तय गिनती के लिए डिफ़ॉल्ट वैल्यू भी सेट की जाती हैं
समर्थित हैं. जब कोई डिफ़ॉल्ट वैल्यू तय नहीं की गई हो, तो 0-लाइक या खाली वैल्यू का इस्तेमाल किया जाता है.
बिना किसी डिफ़ॉल्ट वैल्यू वाली कैलकुलेशन, 0 पर शुरू होती हैं. ऐसा तब भी होता है, जब
शून्य एन्यूमरेटर नहीं है.
स्टब लाइब्रेरी का इस्तेमाल करना
अपने मॉड्यूल पर डिपेंडेंसी के तौर पर स्टब लाइब्रेरी जोड़ने के बाद, आपको
उन्हें अपनी फ़ाइलों में शामिल कर सकता है. यहां स्टब लाइब्रेरी के उदाहरण दिए गए हैं.
बिल्ड सिस्टम (लेगसी मॉड्यूल की परिभाषाओं के लिए भी Android.mk
का इस्तेमाल किया जा सकता है):
cc_... {
name: ...,
shared_libs: ["my-module-name-cpp"],
...
}
# or
java_... {
name: ...,
// can also be shared_libs if your preference is to load a library and share
// it among multiple users or if you only need access to constants
static_libs: ["my-module-name-java"],
...
}
# or
rust_... {
name: ...,
rustlibs: ["my-module-name-rust"],
...
}
C++ में उदाहरण:
#include "some/package/IFoo.h"
#include "some/package/Thing.h"
...
// use just like traditional AIDL
Java में उदाहरण:
import some.package.IFoo;
import some.package.Thing;
...
// use just like traditional AIDL
Rust में उदाहरण:
use aidl_interface_name::aidl::some::package::{IFoo, Thing};
...
// use just like traditional AIDL
वर्शन बनाने के लिए इंटरफ़ेस
foo नाम वाले मॉड्यूल का एलान करने से बिल्ड सिस्टम में भी टारगेट बन जाता है
जिसका इस्तेमाल करके मॉड्यूल के एपीआई को मैनेज किया जा सकता है. foo-freeze-api बनाने के बाद
api_dir
या
Android वर्शन के हिसाब से aidl_api/name
, और
एक .hash
फ़ाइल जोड़ता है, जो दोनों फ़ाइल फ़ॉर्मैट के नए फ़्रीज़ किए गए वर्शन को
इंटरफ़ेस पर कॉपी करने की सुविधा मिलती है. foo-freeze-api की मदद से भी versions_with_info
प्रॉपर्टी को अपडेट किया जा सकता है
अतिरिक्त वर्शन और imports
वर्शन दिखाने के लिए. बुनियादी तौर पर,
versions_with_info
की imports
को imports
फ़ील्ड से कॉपी किया गया. हालांकि,
इसके लिए versions_with_info
में imports
में सबसे नए स्टेबल वर्शन की जानकारी दी गई है
आयात करें, जिसका कोई स्पष्ट वर्शन नहीं है.
versions_with_info
प्रॉपर्टी तय होने के बाद, बिल्ड सिस्टम काम करता है
फ़्रीज़ किए गए वर्शन और 'टॉप ऑफ़ ट्री' (ToT) के बीच भी कंपैटबिलिटी की जांच
साथ ही, फ़्रीज़ किया गया नया वर्शन भी मौजूद रहेगा.
इसके अलावा, आपको ToT वर्शन की एपीआई डेफ़िनिशन को भी मैनेज करना होगा. जब भी कोई एपीआई
अपडेट किया गया, अपडेट करने के लिए foo-update-api चलाएं
aidl_api/name/current
इसमें ToT वर्शन की एपीआई परिभाषा शामिल है.
इंटरफ़ेस को एक जैसा बनाए रखने के लिए, मालिक इसमें ये नई चीज़ें जोड़ सकते हैं:
- इंटरफ़ेस के आखिर तक के तरीके या साफ़ तौर पर नए के बारे में बताने वाले नए तरीके सीरियल)
- पार्स किए जा सकने वाले एलिमेंट के आखिर में मौजूद एलिमेंट (हर एलिमेंट के लिए डिफ़ॉल्ट वैल्यू जोड़ना ज़रूरी है एलिमेंट)
- कॉन्स्टेंट वैल्यू
- Android 11 में एन्यूमरेटर
- Android 12 में, यूनियन के आखिर तक के फ़ील्ड
किसी और कार्रवाई की अनुमति नहीं है और न ही कोई और इंटरफ़ेस में बदलाव कर सकता है (ऐसा न करने पर, मालिक के किए गए बदलावों का असर पड़ सकता है).
सभी इंटरफ़ेस रिलीज़ करने के लिए फ़्रीज़ किए गए हैं या नहीं, इसकी जांच करने के लिए पर्यावरण से जुड़े कौनसे वैरिएबल सेट किए गए हैं:
AIDL_FROZEN_REL=true m ...
- बिल्ड के लिए सभी स्थायी एआईडीएल इंटरफ़ेस ज़रूरी हैं को फ़्रीज़ किया जाना चाहिए, जिसमें कोईowner:
फ़ील्ड तय नहीं किया गया हो.AIDL_FROZEN_OWNERS="aosp test"
- बिल्ड के लिए सभी स्थायी एआईडीएल इंटरफ़ेस ज़रूरी हैं "aosp" के तौर पर बताए गएowner:
फ़ील्ड के साथ फ़्रीज़ करने के लिए या "टेस्ट" होना चाहिए.
इंपोर्ट होने की स्थिरता
किसी इंटरफ़ेस के फ़्रीज़ किए गए वर्शन के लिए इंपोर्ट वर्शन को अपडेट करना स्टेबल एआईडीएल लेयर के साथ पुराने सिस्टम के साथ काम करने की सुविधा. हालांकि, उन्हें अपडेट करने के लिए इंटरफ़ेस के पिछले वर्शन का इस्तेमाल करने वाले सभी सर्वर और क्लाइंट को अपडेट करना होगा, और कई तरह के वर्शन को मिलाकर इस्तेमाल करने पर, कुछ ऐप्लिकेशन भ्रम की स्थिति में पड़ सकते हैं. आम तौर पर, सिर्फ़ टाइप या सामान्य पैकेज के लिए, यह सुरक्षित होता है, क्योंकि कोड को जो आईपीसी लेन-देन से जुड़े अज्ञात टाइप के मामलों को हैंडल करने के लिए पहले से लिखे गए हों.
Android प्लैटफ़ॉर्म कोड में android.hardware.graphics.common
सबसे बड़ा है
देखें.
वर्शन वाले इंटरफ़ेस का इस्तेमाल करना
इंटरफ़ेस के तरीके
रनटाइम के दौरान, किसी पुराने सर्वर पर नए तरीकों को कॉल करने पर, नए क्लाइंट को कोई गड़बड़ी या अपवाद हो सकता है.
cpp
बैकएंड को::android::UNKNOWN_TRANSACTION
मिलता है.ndk
बैकएंड कोSTATUS_UNKNOWN_TRANSACTION
मिलता है.java
बैकएंड कोandroid.os.RemoteException
मैसेज मिलता है, जिसमें यह लिखा होता है एपीआई लागू नहीं किया गया है.
इसे मैनेज करने वाली रणनीतियों के लिए, यहां देखें क्वेरी वर्शन और डिफ़ॉल्ट का इस्तेमाल कर सकते हैं.
पार्स किए जा सकने वाले
जब पार्स किए जा सकने वाले किसी फ़ील्ड में नए फ़ील्ड जोड़े जाते हैं, तो पुराने क्लाइंट और सर्वर उन्हें छोड़ देते हैं. जब नए क्लाइंट और सर्वर को पुराने पार्स किए जा सकने वाले डेटा मिलते हैं, तो फ़ील्ड अपने-आप भर जाते हैं. इसका मतलब है कि डिफ़ॉल्ट रूप से पार्सल किए जा सकने वाले सभी नए फ़ील्ड के लिए तय करें.
क्लाइंट को यह उम्मीद नहीं करनी चाहिए कि सर्वर नए फ़ील्ड का इस्तेमाल तब तक करेंगे, जब तक कि उन्हें सर्वर ऐसा वर्शन लागू कर रहा है जिसमें फ़ील्ड परिभाषित है (देखें क्वेरी वर्शन).
Enum और कॉन्सटेंट
इसी तरह, क्लाइंट और सर्वर को अनजाने में अस्वीकार या अनदेखा कर देना चाहिए स्थिर मान और एन्यूमरेटर, जैसा कि उचित हो, क्योंकि इसमें और भी जोड़ा जा सकता है आने वाले समय में. उदाहरण के लिए, सर्वर को enumerator है जिसके बारे में उसे कोई जानकारी नहीं है. सर्वर को या तो enumerator का इस्तेमाल करें या कोई ऐसी चीज़ लौटाए जिससे क्लाइंट को पता चल जाए कि वह इसमें काम नहीं करता को लागू किया जा सकता है.
यूनियन
प्राप्तकर्ता पुराना होने पर नए फ़ील्ड के साथ यूनियन भेजने का प्रयास विफल हो जाता है और
फ़ील्ड के बारे में नहीं जानता. लागू करने पर
क्लिक करें. इस गड़बड़ी को अनदेखा किया जाता है, अगर
एकतरफ़ा लेन-देन; ऐसा न होने पर, गड़बड़ी BAD_VALUE
(C++ या NDK के लिए) होगी
बैकएंड) या IllegalArgumentException
(Java बैकएंड के लिए उपलब्ध है). गड़बड़ी यह है
तभी प्राप्त होता है, जब ग्राहक नए फ़ील्ड में यूनियन सेट को किसी पुराने
सर्वर या जब यह कोई पुराना क्लाइंट हो और नए सर्वर से यूनियन को प्राप्त हो रहा हो.
फ़्लैग-आधारित डेवलपमेंट
रिलीज़ डिवाइसों पर इन-डेवलपमेंट (अनफ़्रीज़ किए गए) इंटरफ़ेस इस्तेमाल नहीं किए जा सकते, क्योंकि हालांकि, पुराने सिस्टम के साथ काम करने की कोई गारंटी नहीं होती.
एआईडीएल, फ़्रीज़ न की गई इन इंटरफ़ेस लाइब्रेरी के लिए रन टाइम फ़ॉलबैक के साथ काम करता है, ताकि ताकि कोड को नए फ़्रीज़ किए गए वर्शन के हिसाब से लिखा जा सके और फिर भी उसका इस्तेमाल किया जा सके रिलीज़ डिवाइस पर. क्लाइंट का पुराने सिस्टम के साथ काम करने का तरीका मौजूदा व्यवहार और फ़ॉलबैक के साथ, लागू करने के तरीके का नहीं है. यहां जाएं: वर्शन वाले इंटरफ़ेस का इस्तेमाल करना.
एआईडीएल बिल्ड फ़्लैग
इस व्यवहार को कंट्रोल करने वाला फ़्लैग RELEASE_AIDL_USE_UNFROZEN
है
build/release/build_flags.bzl
में परिभाषित किया गया है. true
का मतलब है, इसका ऐसा वर्शन जो फ़्रीज़ नहीं किया गया है
इंटरफ़ेस का इस्तेमाल रन टाइम पर किया जाता है और false
का मतलब है,
जिन वर्शन को फ़्रीज़ नहीं किया जाता है वे अपने पिछले फ़्रीज़ किए गए वर्शन की तरह काम करते हैं.
आप इसके लिए फ़्लैग को true
में बदल सकते हैं
लेकिन रिलीज़ से पहले इसे false
में वापस लाना होगा. आम तौर पर
डेवलपमेंट एक ऐसे कॉन्फ़िगरेशन के साथ पूरा किया जाता है, जिसमें फ़्लैग true
पर सेट होता है.
कंपैटबिलिटी मैट्रिक्स और मेनिफ़ेस्ट
वेंडर इंटरफ़ेस ऑब्जेक्ट (VINTF ऑब्जेक्ट) के बारे में जानकारी रिपोर्ट में कौनसे वर्शन की उम्मीद की जाती है और दोनों साइड पर कौनसे वर्शन उपलब्ध कराए जाते हैं वेंडर इंटरफ़ेस पर काम करती है.
ज़्यादातर नॉन-कटलफ़िश डिवाइस, नए वर्शन के साथ काम करने वाले मैट्रिक्स को टारगेट करते हैं
इंटरफ़ेस के फ़्रीज़ होने के बाद ही, एआईडीएल में कोई अंतर नहीं होता
RELEASE_AIDL_USE_UNFROZEN
पर आधारित लाइब्रेरी.
मैट्रिक्स
पार्टनर के मालिकाना हक वाले इंटरफ़ेस, डिवाइस या किसी प्रॉडक्ट से जुड़े इंटरफ़ेस पर जोड़े जाते हैं
कंपैटबिलिटी मैट्रिक्स, जिन्हें डिवाइस डेवलपमेंट के दौरान टारगेट करता है. इसलिए, जब किसी
कंपैटबिलिटी मैट्रिक्स में, किसी इंटरफ़ेस का नया और फ़्रीज़ न किया गया वर्शन जोड़ा गया है,
पिछले फ़्रीज़ किए गए वर्शन
RELEASE_AIDL_USE_UNFROZEN=false
. तो अलग-अलग
अलग-अलग RELEASE_AIDL_USE_UNFROZEN
के साथ काम करने वाली मैट्रिक्स फ़ाइलें
कॉन्फ़िगरेशन या एक ही कंपैटबिलिटी मैट्रिक्स फ़ाइल में दोनों वर्शन की अनुमति देना
इसका इस्तेमाल सभी कॉन्फ़िगरेशन में किया जाएगा.
उदाहरण के लिए, फ़्रीज़ न किए गए वर्शन 4 को जोड़ते समय, <version>3-4</version>
का इस्तेमाल करें.
वर्शन 4 के फ़्रीज़ होने पर, काम करने वाले मैट्रिक्स से वर्शन 3 को हटाया जा सकता है
क्योंकि फ़्रीज़ किए गए वर्शन 4 का इस्तेमाल तब किया जाता है, जब RELEASE_AIDL_USE_UNFROZEN
false
.
मेनिफ़ेस्ट
Android 15 (एओएसपी एक्सपेरिमेंट) में, libvintf
में बदलाव किया गया है
बिल्ड के समय मेनिफ़ेस्ट फ़ाइलों में बदलाव
RELEASE_AIDL_USE_UNFROZEN
.
मेनिफ़ेस्ट और मेनिफ़ेस्ट फ़्रैगमेंट बताते हैं कि इंटरफ़ेस का कौनसा वर्शन
लागू होता है. किसी इंटरफ़ेस के फ़्रीज़ न किए गए नए वर्शन का इस्तेमाल करते समय,
इस नए वर्शन को दिखाने के लिए मेनिफ़ेस्ट को अपडेट करना होगा. टास्क कब शुरू होगा
RELEASE_AIDL_USE_UNFROZEN=false
मेनिफ़ेस्ट एंट्री में बदलाव
जनरेट की गई एआईडीएल लाइब्रेरी में हुए बदलाव को दिखाने के लिए, libvintf
. वर्शन
को फ़्रीज़ नहीं किए गए वर्शन N
से बदला गया है और
फ़्रीज़ किया गया पिछला वर्शन N - 1
था. इसलिए, उपयोगकर्ताओं को कई विज्ञापनों
अपनी हर सेवा के लिए मेनिफ़ेस्ट या मेनिफ़ेस्ट फ़्रैगमेंट.
एचएएल क्लाइंट में किए गए बदलाव
HAL क्लाइंट कोड, हर बार काम करने वाले पिछले फ़्रीज़ किए गए फ़ंक्शन के साथ पुराने सिस्टम के साथ काम करना चाहिए
वर्शन है. जब RELEASE_AIDL_USE_UNFROZEN
false
हो, तो सेवाएं हमेशा दिखती हैं
जैसे, पिछली बार फ़्रीज़ किया गया वर्शन या उससे पहले का वर्शन (उदाहरण के लिए, नए सॉफ़्टवेयर को फ़्रीज़ न करने की सुविधा के साथ कॉल करना)
तरीकों से UNKNOWN_TRANSACTION
मिलता है या parcelable
के नए फ़ील्ड में
डिफ़ॉल्ट वैल्यू). Android फ़्रेमवर्क क्लाइंट का बैक अप लेना ज़रूरी है
पिछले वर्शन के साथ काम करता है, लेकिन यह
पार्टनर के मालिकाना हक वाले इंटरफ़ेस के वेंडर क्लाइंट और क्लाइंट.
एचएएल को लागू करने के तरीके में बदलाव
झंडे पर आधारित डेवलपमेंट के साथ एचएएल डेवलपमेंट में सबसे बड़ा अंतर यह है कि
एचएएल को लागू करने के लिए ज़रूरी है कि वह पिछले एपीआई के साथ पुराने सिस्टम के साथ काम करे
RELEASE_AIDL_USE_UNFROZEN
के false
होने पर, फ़्रीज़ किया गया वर्शन काम करेगा.
डिवाइस कोड और लागू करने के तरीके में पुराने सिस्टम के साथ काम करने की सुविधा को ध्यान में रखना
व्यायाम. वर्शन वाले वर्शन का इस्तेमाल करें को देखें
इंटरफ़ेस में बदल सकते हैं.
पुराने सिस्टम के साथ काम करने से जुड़ी ज़रूरी शर्तें आम तौर पर, क्लाइंट और सर्वर, और फ़्रेमवर्क कोड और वेंडर कोड के लिए उपलब्ध हैं, लेकिन छोटे-छोटे अंतर की जानकारी होनी चाहिए, क्योंकि अब आप असरदार तरीके से एक ही सोर्स कोड का इस्तेमाल करने वाले दो वर्शन लागू करना (मौजूदा, अनफ़्रीज़्ड वर्शन).
उदाहरण: किसी इंटरफ़ेस में तीन फ़्रीज़ किए गए वर्शन होते हैं. इंटरफ़ेस को इससे अपडेट किया जाता है:
नया तरीका इस्तेमाल करें. क्लाइंट और सेवा, दोनों को नए वर्शन 4 का इस्तेमाल करने के लिए अपडेट किया जाता है
लाइब्रेरी. क्योंकि V4 लाइब्रेरी,
इंटरफ़ेस, वह पिछले रुका हुआ वर्शन, वर्शन 3 की तरह काम करता है, जब
RELEASE_AIDL_USE_UNFROZEN
, false
है और नए तरीके के इस्तेमाल को रोकता है.
इंटरफ़ेस के फ़्रीज़ होने पर, RELEASE_AIDL_USE_UNFROZEN
की सभी वैल्यू उसका इस्तेमाल करती हैं
वर्शन को फ़्रीज़ किया जा सकता है और पुराने सिस्टम के साथ काम करने की सुविधा को मैनेज करने वाले कोड को हटाया जा सकता है.
कॉलबैक पर तरीकों को कॉल करते समय, आपको
UNKNOWN_TRANSACTION
लौटाया गया. क्लाइंट शायद दो अलग-अलग डाइमेंशन लागू कर रहे हैं
रिलीज़ कॉन्फ़िगरेशन के आधार पर कॉलबैक के वर्शन प्रदान करता है, इसलिए आप
यह मान लेने के बाद कि क्लाइंट सबसे नया वर्शन भेजता है.
यह. यह ठीक वैसे ही है जैसे स्थायी एआईडीएल क्लाइंट अपने व्यवहार को पीछे की ओर बनाए रखते हैं
सर्वर के साथ काम करता है या नहीं, इसके बारे में वर्शन वाले वर्शन का इस्तेमाल करें
इंटरफ़ेस में बदल सकते हैं.
// Get the callback along with the version of the callback
ScopedAStatus RegisterMyCallback(const std::shared_ptr<IMyCallback>& cb) override {
mMyCallback = cb;
// Get the version of the callback for later when we call methods on it
auto status = mMyCallback->getInterfaceVersion(&mMyCallbackVersion);
return status;
}
// Example of using the callback later
void NotifyCallbackLater() {
// From the latest frozen version (V2)
mMyCallback->foo();
// Call this method from the unfrozen V3 only if the callback is at least V3
if (mMyCallbackVersion >= 3) {
mMyCallback->bar();
}
}
मौजूदा टाइप के नए फ़ील्ड (parcelable
, enum
, union
) में
मौजूद नहीं होता या उसके डिफ़ॉल्ट मान नहीं होते, जब RELEASE_AIDL_USE_UNFROZEN
false
और सेवा जिन नए फ़ील्ड को भेजने की कोशिश करती है उनकी वैल्यू छोड़ दी जाती हैं
हमें कुछ नया करने की ज़रूरत नहीं है.
फ़्रीज़ न किए गए इस वर्शन में जोड़े गए नए टाइप नहीं भेजे जा सकते जिसे इंटरफ़ेस से ऐक्सेस किया जा सकता है.
लागू करने की प्रक्रिया को कभी भी किसी भी क्लाइंट से नए तरीकों के लिए कॉल नहीं किया जाता
RELEASE_AIDL_USE_UNFROZEN
false
है.
नए एन्यूमरेटर का इस्तेमाल सिर्फ़ उस वर्शन के साथ करें जिसमें उन्हें पेश किया गया है, न कि पिछला वर्शन.
आम तौर पर, रिमोट का वर्शन देखने के लिए foo->getInterfaceVersion()
का इस्तेमाल किया जाता है
इंटरफ़ेस का उपयोग कर रहा है. हालांकि फ़्लैग-आधारित वर्शन के साथ, आप
दो अलग-अलग वर्शन लागू करना है, तो हो सकता है कि आप
मौजूदा इंटरफ़ेस पर कॉपी करता है. आप इसका इंटरफ़ेस वर्शन प्राप्त करके ऐसा कर सकते हैं:
मौजूदा ऑब्जेक्ट, उदाहरण के लिए, this->getInterfaceVersion()
या दूसरा
my_ver
की गिनती करने का तरीका. रिमोट के इंटरफ़ेस वर्शन के बारे में क्वेरी करना देखें
ऑब्जेक्ट
हमारा वीडियो देखें.
नए VINTF और स्टेबल इंटरफ़ेस
जब कोई नया AIDL इंटरफ़ेस पैकेज जोड़ा जाता है, तो आखिरी बार फ़्रीज़ किया गया कोई वर्शन नहीं होता, इसलिए
RELEASE_AIDL_USE_UNFROZEN
के होने के बाद वापस आने का कोई व्यवहार नहीं है
false
. इन इंटरफ़ेस का इस्तेमाल न करें. जब RELEASE_AIDL_USE_UNFROZEN
होता है
false
, सेवा मैनेजर, सेवा को इंटरफ़ेस रजिस्टर करने की अनुमति नहीं देगा
और क्लाइंट इसे नहीं खोज पाते.
आप
डिवाइस में मौजूद मेकफ़ाइल में RELEASE_AIDL_USE_UNFROZEN
फ़्लैग:
ifeq ($(RELEASE_AIDL_USE_UNFROZEN),true)
PRODUCT_PACKAGES += \
android.hardware.health.storage-service
endif
अगर सेवा किसी बड़ी प्रोसेस का हिस्सा है, तो आप उसे डिवाइस में नहीं जोड़ सकते
कुछ शर्तों के साथ, यह देखा जा सकता है कि सेवा का एलान
IServiceManager::isDeclared()
. अगर इसका एलान किया गया है और रजिस्टर नहीं किया जा सका, तो
प्रोसेस को रद्द करते हैं. अगर इसका एलान नहीं किया गया है, तो हो सकता है कि वह रजिस्टर न हो पाए.
डेवलपमेंट टूल के तौर पर कटलफ़िश
वीआईएनटीएफ़ के फ़्रीज़ होने के बाद, हर साल हम फ़्रेमवर्क के साथ काम करने की सुविधा में बदलाव करते हैं
मैट्रिक्स (FCM) target-level
और कटलफ़िश का PRODUCT_SHIPPING_API_LEVEL
ताकि ये अगले साल की रिलीज़ में लॉन्च होने वाले डिवाइसों की जानकारी दें. हम एडजस्ट करते हैं
target-level
और PRODUCT_SHIPPING_API_LEVEL
को पक्का करने के लिए कि कुछ
लॉन्च होने वाला ऐसा डिवाइस जो टेस्ट किया जा चुका है और अगले साल के
रिलीज़.
जब RELEASE_AIDL_USE_UNFROZEN
की उम्र true
होती है, तो कटलफ़िश यह होती है
का इस्तेमाल, आने वाले समय में Android के रिलीज़ होने वाले वर्शन को डेवलप करने के लिए किया जाता है. इसका टारगेट Android के लिए, अगले साल तक
रिलीज़ का FCM लेवल और PRODUCT_SHIPPING_API_LEVEL
है, तो उसे पूरा करना ज़रूरी है
वेंडर सॉफ़्टवेयर से जुड़ी ज़रूरी शर्तें (वीएसआर).
जब RELEASE_AIDL_USE_UNFROZEN
की वैल्यू false
है, तब कटलफ़िश के पास पहले वाले नतीजे होते हैं
रिलीज़ डिवाइस को दिखाने के लिए target-level
और PRODUCT_SHIPPING_API_LEVEL
.
Android 14 और उससे पहले के वर्शन में, यह फ़र्क़ सिर्फ़
अलग-अलग Git शाखाओं के साथ पूरा किया जाता है जो FCM में बदलाव नहीं करते
target-level
, शिपिंग एपीआई लेवल या अगले कोड को टारगेट करने वाला कोई दूसरा कोड
रिलीज़.
मॉड्यूल का नाम रखने के नियम
Android 11 में, वर्शन और
तो एक स्टब लाइब्रेरी मॉड्यूल अपने-आप बन जाता है. रेफ़र करने के लिए
किसी खास स्टब लाइब्रेरी मॉड्यूल से लिंक करना है, तो
aidl_interface
मॉड्यूल, लेकिन स्टब लाइब्रेरी मॉड्यूल का नाम, जो कि
ifacename-version-backend, जहां
ifacename
:aidl_interface
मॉड्यूल का नामversion
इनमें से कोई एक है- फ़्रीज़ किए गए वर्शन के लिए
Vversion-number
Vlatest-frozen-version-number + 1
टिप-ऑफ़-ट्री (अभी-अभी फ़्रीज़ होना) वर्शन
- फ़्रीज़ किए गए वर्शन के लिए
backend
इनमें से कोई एक है- Java बैकएंड के लिए
java
, - C++ बैकएंड के लिए
cpp
, - NDK बैकएंड के लिए,
ndk
याndk_platform
. पहली इमेज ऐप्लिकेशन के लिए है और बाद वाला विकल्प, Android 13 तक के लिए प्लैटफ़ॉर्म का इस्तेमाल करने के लिए है. तय सीमा में Android 13 और उसके बाद के वर्शन में, सिर्फ़ndk
का इस्तेमाल किया जा सकता है. - Rust बैकएंड के लिए
rust
.
- Java बैकएंड के लिए
मान लें कि foo नाम वाला एक मॉड्यूल है और उसका सबसे नया वर्शन 2 है, और यह NDK और C++, दोनों पर काम करता है. इस मामले में, एआईडीएल ये मॉड्यूल जनरेट करता है:
- वर्शन 1 के मुताबिक
foo-V1-(java|cpp|ndk|ndk_platform|rust)
- वर्शन 2 (नया स्टेबल वर्शन) पर आधारित
foo-V2-(java|cpp|ndk|ndk_platform|rust)
- सेवा की शर्तों (ToT) के वर्शन के मुताबिक
foo-V3-(java|cpp|ndk|ndk_platform|rust)
Android 11 की तुलना में:
foo-backend
, जिसे नए स्टेबल ऑडियो के तौर पर जाना जाता है वर्शनfoo-V2-backend
हो जाता हैfoo-unstable-backend
, यह सेवा की शर्तों के बारे में है वर्शनfoo-V3-backend
हो जाता है
आउटपुट फ़ाइल के नाम हमेशा मॉड्यूल के नाम जैसे होते हैं.
- वर्शन 1 पर आधारित:
foo-V1-(cpp|ndk|ndk_platform|rust).so
- वर्शन 2 पर आधारित:
foo-V2-(cpp|ndk|ndk_platform|rust).so
- सेवा की शर्तों (ToT) के वर्शन पर आधारित:
foo-V3-(cpp|ndk|ndk_platform|rust).so
ध्यान दें कि एआईडीएल कंपाइलर, unstable
वर्शन वाला मॉड्यूल नहीं बनाता है,
या स्थिर एआईडीएल इंटरफ़ेस के लिए बिना वर्शन वाला मॉड्यूल.
Android 12 के बाद से, मॉड्यूल का नाम
स्थायी AIDL इंटरफ़ेस में हमेशा इसका वर्शन शामिल होता है.
मेटा इंटरफ़ेस की नई विधियां
Android 10, स्थिर एआईडीएल.
रिमोट ऑब्जेक्ट के इंटरफ़ेस वर्शन की क्वेरी करें
क्लाइंट, उस इंटरफ़ेस के वर्शन और हैश से क्वेरी कर सकते हैं जो रिमोट ऑब्जेक्ट इंटरफ़ेस में दिखाई गई वैल्यू को लागू करके, दिखाई गई वैल्यू की तुलना इंटरफ़ेस में मौजूद वैल्यू से करता है इस्तेमाल कर रहा है.
cpp
बैकएंड के साथ उदाहरण:
sp<IFoo> foo = ... // the remote object
int32_t my_ver = IFoo::VERSION;
int32_t remote_ver = foo->getInterfaceVersion();
if (remote_ver < my_ver) {
// the remote side is using an older interface
}
std::string my_hash = IFoo::HASH;
std::string remote_hash = foo->getInterfaceHash();
ndk
(और ndk_platform
) बैकएंड के साथ उदाहरण:
IFoo* foo = ... // the remote object
int32_t my_ver = IFoo::version;
int32_t remote_ver = 0;
if (foo->getInterfaceVersion(&remote_ver).isOk() && remote_ver < my_ver) {
// the remote side is using an older interface
}
std::string my_hash = IFoo::hash;
std::string remote_hash;
foo->getInterfaceHash(&remote_hash);
java
बैकएंड के साथ उदाहरण:
IFoo foo = ... // the remote object
int myVer = IFoo.VERSION;
int remoteVer = foo.getInterfaceVersion();
if (remoteVer < myVer) {
// the remote side is using an older interface
}
String myHash = IFoo.HASH;
String remoteHash = foo.getInterfaceHash();
Java की भाषा के लिए, रिमोट साइड को getInterfaceVersion()
और
getInterfaceHash()
का इस्तेमाल इस तरह करें (इनसे बचने के लिए, IFoo
के बजाय super
का इस्तेमाल किया जाता है
कॉपी करने और चिपकाने में हुई गलतियां. @SuppressWarnings("static")
यह एनोटेशन कर सकता है
javac
कॉन्फ़िगरेशन के आधार पर, चेतावनियां बंद करने की ज़रूरत होगी):
class MyFoo extends IFoo.Stub {
@Override
public final int getInterfaceVersion() { return super.VERSION; }
@Override
public final String getInterfaceHash() { return super.HASH; }
}
इसकी वजह यह है कि जनरेट की गई क्लास (IFoo
, IFoo.Stub
वगैरह) शेयर की गई हैं
और क्लाइंट और सर्वर के बीच सेट कर सकते हैं (उदाहरण के लिए, क्लास बूट में
क्लास पाथ). जब क्लास शेयर की जाती हैं, तो सर्वर को
क्लास का सबसे नया वर्शन दिया है, भले ही इसे किसी पुराने ज़माने की जानकारी के साथ बनाया गया हो
एक वर्शन है. अगर यह मेटा इंटरफ़ेस शेयर किए गए
क्लास है, तो यह हमेशा सबसे नया वर्शन लौटाता है. हालांकि, डाइग्नोस्टिक टूल का इस्तेमाल करके
जैसा कि ऊपर बताया गया है, इंटरफ़ेस का वर्शन नंबर सर्वर के कोड में एम्बेड होता है
(क्योंकि IFoo.VERSION
एक static final int
है, जिसे रेफ़रंस के समय इनलाइन किया जाता है)
इस तरह, यह तरीका उसी वर्शन को दिखा सकता है जिससे सर्वर बनाया गया था.
पुराने इंटरफ़ेस से निपटें
ऐसा हो सकता है कि क्लाइंट को एआईडीएल के नए वर्शन के साथ अपडेट किया गया हो
इंटरफ़ेस है लेकिन सर्वर पुराने एआईडीएल इंटरफ़ेस का इस्तेमाल कर रहा है. ऐसे मामलों में,
किसी पुराने इंटरफ़ेस पर किसी तरीके को कॉल करने से UNKNOWN_TRANSACTION
नतीजा मिलता है.
एआईडीएल के साथ बेहतर तरीके से काम करने पर, क्लाइंट को ज़्यादा कंट्रोल मिलता है. क्लाइंट साइड में, आपके पास ये विकल्प सेट करने का विकल्प होता है एआईडीएल इंटरफ़ेस पर डिफ़ॉल्ट तौर पर लागू होता है. डिफ़ॉल्ट तरीका लागू करने की प्रोसेस सिर्फ़ तब लागू की जाती है, जब रिमोट में तरीका लागू न किया गया हो कहा जाता है (क्योंकि इसे इंटरफ़ेस के पुराने वर्शन के साथ बनाया गया था). से डिफ़ॉल्ट सेटिंग वैश्विक रूप से सेट हैं, उनका उपयोग संभावित रूप से शेयर किए गए संदर्भ.
Android 13 और उसके बाद के वर्शन में C++ पर इसका उदाहरण:
class MyDefault : public IFooDefault {
Status anAddedMethod(...) {
// do something default
}
};
// once per an interface in a process
IFoo::setDefaultImpl(::android::sp<MyDefault>::make());
foo->anAddedMethod(...); // MyDefault::anAddedMethod() will be called if the
// remote side is not implementing it
Java में उदाहरण:
IFoo.Stub.setDefaultImpl(new IFoo.Default() {
@Override
public xxx anAddedMethod(...) throws RemoteException {
// do something default
}
}); // once per an interface in a process
foo.anAddedMethod(...);
आपको एआईडीएल में सभी तरीकों को डिफ़ॉल्ट रूप से लागू करने की सुविधा देने की ज़रूरत नहीं है
इंटरफ़ेस पर कॉपी करने की सुविधा मिलती है. ऐसे तरीके जिन्हें रिमोट साइड में लागू किए जाने की गारंटी होती है
(क्योंकि आपको इस बात पर यकीन है कि रिमोट तब बनाया गया है, जब
एआईडीएल इंटरफ़ेस की जानकारी) को डिफ़ॉल्ट impl
में बदलने की ज़रूरत नहीं होती
क्लास.
मौजूदा एआईडीएल को स्ट्रक्चर्ड या स्टेबल एआईडीएल में बदलें
अगर आपके पास कोई मौजूदा एआईडीएल इंटरफ़ेस और इस्तेमाल करने वाला कोड है, तो इनका इस्तेमाल करें इस तरीके का इस्तेमाल करके इंटरफ़ेस को स्थिर एआईडीएल इंटरफ़ेस में बदला जा सकता है.
अपने इंटरफ़ेस की सभी निर्भरताओं का पता लगाएं. हर पैकेज के लिए इंटरफ़ेस इस बात पर निर्भर करता है कि पैकेज स्टेबल एआईडीएल में है या नहीं. अगर आपने परिभाषित नहीं है, तो पैकेज को रूपांतरित किया जाना चाहिए.
अपने इंटरफ़ेस में मौजूद सभी पार्सल को स्टेबल पार्सेबल में बदलें ( इंटरफ़ेस फ़ाइलों में कोई बदलाव नहीं होता). इस तरीके से करें एआईडीएल फ़ाइलों में अपना स्ट्रक्चर सीधे तौर पर दिखाना होता है. मैनेजमेंट क्लास को यह ज़रूरी है इन नए प्रकारों का इस्तेमाल करने के लिए उन्हें फिर से लिखा जा सकता है. यह कार्रवाई करने से पहले किया जा सकता है
aidl_interface
पैकेज (नीचे).ऐसा
aidl_interface
पैकेज बनाएं (जैसा कि ऊपर बताया गया है) जिसमें आपके मॉड्यूल का नाम, उसकी डिपेंडेंसी, और आपकी ज़रूरत की कोई भी अन्य जानकारी. इसे स्टेबलाइज़ करने के लिए (सिर्फ़ स्ट्रक्चर ही नहीं) करने के लिए, इसका वर्शन भी होना चाहिए. ज़्यादा जानकारी के लिए, वर्शन इंटरफ़ेस देखें.