ב-Android 10 נוספה תמיכה ב-Android Interface Definition Language (AIDL) יציב, דרך חדשה לעקוב אחרי ממשק תכנות האפליקציה (API) וממשק האפליקציה הבינארי (ABI) שסופקו על ידי ממשקי AIDL. Stable AIDL פועל בדיוק כמו AIDL, אבל מערכת ה-build עוקבת אחרי תאימות הממשק, ויש הגבלות על מה שאפשר לעשות:
- ממשקים מוגדרים במערכת ה-build באמצעות
aidl_interfaces
. - ממשקים יכולים להכיל רק נתונים מובְנים. אובייקטים מסוג Parcelable שמייצגים את הסוגים המועדפים נוצרים באופן אוטומטי על סמך ההגדרה שלהם ב-AIDL, ומתבצעים להם אוטומטית תהליכי marshalling ו-unmarshalling.
- אפשר להצהיר על ממשקים כיציבים (תואמים לאחור). במקרה כזה, המערכת עוקבת אחרי ממשק ה-API ומעדכנת את הגרסה שלו בקובץ לצד ממשק ה-AIDL.
יצירת גרסת AIDL מובנית לעומת יצירת גרסת AIDL יציבה
Structured AIDL מתייחס לסוגים שמוגדרים רק ב-AIDL. לדוגמה, הצהרת parcelable (parcelable בהתאמה אישית) היא לא AIDL מובנה. אובייקטים מסוג Parcelable עם השדות שלהם מוגדרים ב-AIDL נקראים Parcelables מובְנים.
כדי ליצור קובצי AIDL יציבים, צריך ליצור קובצי AIDL מובְנים כדי שמערכת ה-build והמְהַדר יוכלו להבין אם השינויים שבוצעו ב-Parcelables תואמים לאחור.
עם זאת, לא כל הממשקים המובְנים יציבים. כדי שהממשק יהיה יציב, צריך להשתמש בו רק בסוגי נתונים מובְנים, וגם בתכונות הבאות של ניהול גרסאות. לעומת זאת, ממשק לא יציב אם נעשה בו שימוש במערכת ה-build של הליבה או אם הערך unstable:true
מוגדר.
הגדרת ממשק AIDL
ההגדרה של 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
: השם של מודול ממשק ה-AIDL שמזהה באופן ייחודי ממשק AIDL.srcs
: רשימת קובצי המקור של AIDL שמרכיבים את הממשק. הנתיב של סוג AIDLFoo
שמוגדר בחבילהcom.acme
צריך להיות ב-<base_path>/com/acme/Foo.aidl
, כאשר<base_path>
יכול להיות כל ספרייה שקשורה לספרייה שבה נמצאAndroid.bp
. בדוגמה הקודמת,<base_path>
הואsrcs/aidl
.local_include_dir
: הנתיב שממנו מתחיל שם החבילה. הוא תואם ל-<base_path>
שמתואר למעלה.imports
: רשימה של המודולים שלaidl_interface
שבהם נעשה שימוש. אם אחד מממשקי ה-AIDL שלכם משתמש בממשק או ב-Parcelable מ-aidl_interface
אחר, מציינים כאן את השם שלו. אפשר להשתמש בשם לבד כדי להתייחס לגרסה האחרונה, או בשם עם הסיומת של הגרסה (כמו-V1
) כדי להתייחס לגרסה ספציפית. אפשר לציין גרסה החל מגרסה Android 12versions
: הגרסאות הקודמות של הממשק שמושהות ב-api_dir
. החל מ-Android 11, ה-versions
מושהים ב-aidl_api/name
. אם אין גרסאות קפואות של ממשק, לא צריך לציין זאת ולא יתבצעו בדיקות תאימות. השדה הזה הוחלף ב-versions_with_info
ב-Android מגרסה 13 ואילך.versions_with_info
: רשימה של צמדי מחרוזות (tuples), שכל אחד מהם מכיל את השם של גרסת קריאו ורשימה של גרסאות ייבוא של מודולים אחרים של aidl_interface שגרסת aidl_interface הזו ייבאה. ההגדרה של הגרסה V של ממשק AIDL IFACE נמצאת בכתובתaidl_api/IFACE/V
. השדה הזה הוצג ב-Android 13, ואין לשנות אותו ישירות ב-Android.bp
. כדי להוסיף או לעדכן את השדה, קוראים לפונקציה*-update-api
או*-freeze-api
. כמו כן, שדותversions
מועברים באופן אוטומטי אלversions_with_info
כשמשתמש מפעיל את*-update-api
או את*-freeze-api
.stability
: הדגל האופציונלי של ההתחייבות ליציבות של הממשק הזה. האפשרות הזו תומכת רק ב-"vintf"
. אם לא מגדירים אתstability
, מערכת ה-build בודקת שהממשק תואם לאחור, אלא אם מציינים אתunstable
. מצב 'לא מוגדר' תואם לממשק עם יציבות בהקשר של הידור הזה (כלומר, כל הדברים במערכת, למשל דברים ב-system.img
ובמחיצות קשורות, או כל הדברים של הספק, למשל דברים ב-vendor.img
ובמחיצות קשורות). אם הערך שלstability
מוגדר כ-"vintf"
, המשמעות היא התחייבות ליציבות: הממשק חייב להישאר יציב כל עוד משתמשים בו.gen_trace
: הדגל האופציונלי להפעלה או להשבתה של המעקב. החל מגרסה Android 14, ברירת המחדל היאtrue
לקצוות העורפיcpp
ו-java
.host_supported
: הדגל האופציונלי. כשמגדירים אותו ל-true
, הספריות שנוצרות זמינות לסביבת המארח.unstable
: הדגל האופציונלי שמשמש לסימון שהממשק הזה לא חייב להיות יציב. כשהערך מוגדר ל-true
, מערכת ה-build לא יוצרת את דמפ ה-API לממשק ולא דורשת לעדכן אותו.frozen
: הדגל האופציונלי. אם הוא מוגדר כ-true
, המשמעות היא שלא בוצעו שינויים בממשק מאז הגרסה הקודמת של הממשק. כך תוכלו לבצע בדיקות נוספות בזמן ה-build. כשהערך מוגדר ל-false
, המשמעות היא שהממשק נמצא בפיתוח ויש בו שינויים חדשים. לכן, הפעלתfoo-freeze-api
יוצרת גרסה חדשה ומשנה את הערך ל-true
באופן אוטומטי. הוצגה ב-Android 14.backend.<type>.enabled
: הדגלים האלה מפעילים או משביתים כל אחד מהקצוות העורפיים שבשבילם מפיק קובצי קוד המהדר של AIDL. יש תמיכה בארבעה קצוות עורפיים: Java, C++, NDK ו-Rust. הקצוות העורפיים של Java, C++ ו-NDK מופעלים כברירת מחדל. אם אתם לא צריכים אף אחד משלושת הקצוות העורפי האלה, צריך להשבית אותו באופן מפורש. Rust מושבת כברירת מחדל עד Android 15.backend.<type>.apex_available
: רשימת השמות של APEX שעבורם ספריית ה-stub שנוצרה זמינה.backend.[cpp|java].gen_log
: הדגל האופציונלי שקובע אם ליצור קוד נוסף לאיסוף מידע על העסקה.backend.[cpp|java].vndk.enabled
: הדגל האופציונלי להפיכת הממשק הזה לחלק מ-VNDK. ברירת המחדל היאfalse
.backend.[cpp|ndk].additional_shared_libraries
: הדגל הזה, שהוצג ב-Android 14, מוסיף יחסי תלות לספריות המקומיות. הדגל הזה שימושי עםndk_header
ו-cpp_header
.backend.java.sdk_version
: הדגל האופציונלי לציון הגרסה של ה-SDK שעבורה נוצרה ספריית ה-stub של Java. ערך ברירת המחדל הוא"system_current"
. לא צריך להגדיר את הערך הזה כשהערך שלbackend.java.platform_apis
הואtrue
.backend.java.platform_apis
: הדגל האופציונלי שצריך להגדיר לערךtrue
כשצריך לבנות את הספריות שנוצרו מול ממשק ה-API של הפלטפורמה במקום מול ה-SDK.
לכל שילוב של הגרסאות והקצוות העורפיים המופעלים, נוצרת ספריית stub. במאמר כללי מתן שמות למערכי מודולים מוסבר איך להפנות לגרסה הספציפית של ספריית ה-stub לקצה עורפי ספציפי.
כתיבת קובצי AIDL
ממשקים ב-AIDL יציב דומים לממשקים רגילים, מלבד העובדה שאסור להשתמש בהם ב-parcelables לא מובנים (כי הם לא יציבים. אפשר לעיין במאמר AIDL מובנה לעומת AIDL יציב). ההבדל העיקרי ב-AIDL היציב הוא האופן שבו מוגדרים רכיבים שניתן לחלק (parcelables). בעבר, הצהרתם מראש על רכיבי Parcelable. ב-AIDL יציב (ולכן מובנה), השדות והמשתנים של רכיבי Parcelable מוגדרים באופן מפורש.
// 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 גם אם אין מונה של אפס.
שימוש בספריות stub
אחרי שמוסיפים ספריות stub כיחסי תלות למודול, אפשר לכלול אותן בקבצים. ריכזנו כאן דוגמאות לספריות stub במערכת ה-build (אפשר להשתמש ב-Android.mk
גם להגדרות של מודולים מדור קודם).
הערה: בדוגמאות האלה, הגרסה לא מופיעה, כך שהן מייצגות שימוש בממשק לא יציב. עם זאת, שמות של ממשקים עם גרסאות כוללים מידע נוסף. אפשר לעיין במאמר ממשקים עם ניהול גרסאות.
cc_... {
name: ...,
// use `shared_libs:` to load your library and its transitive dependencies
// dynamically
shared_libs: ["my-module-name-cpp"],
// use `static_libs:` to include the library in this binary and drop
// transitive dependencies
static_libs: ["my-module-name-cpp"],
...
}
# or
java_... {
name: ...,
// use `static_libs:` to add all jars and classes to this jar
static_libs: ["my-module-name-java"],
// use `libs:` to make these classes available during build time, but
// not add them to the jar, in case the classes are already present on the
// boot classpath (such as if it's in framework.jar) or another jar.
libs: ["my-module-name-java"],
// use `srcs:` with `-java-sources` if you want to add classes in this
// library jar directly, but you get transitive dependencies from
// somewhere else, such as the boot classpath or another jar.
srcs: ["my-module-name-java-source", ...],
...
}
# 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 יוצרת גם יעד במערכת ה-build שאפשר להשתמש בו כדי לנהל את ה-API של המודול. במהלך ה-build, foo-freeze-api מוסיף הגדרת API חדשה בקטע api_dir
או aidl_api/name
, בהתאם לגרסת Android, ומוסיף קובץ .hash
. שניהם מייצגים את הגרסה החדשה והקפואה של הממשק. foo-freeze-api מעדכן גם את המאפיין versions_with_info
כך שישקף את הגרסה הנוספת ואת imports
לגרסה. בעיקרון, הערך של imports
ב-versions_with_info
מועתק מהשדה imports
. אבל הגרסה היציבה האחרונה צוינה ב-imports
ב-versions_with_info
לייבוא, שאין לו גרסה מפורשת.
אחרי שמציינים את המאפיין versions_with_info
, מערכת ה-build מריצה בדיקות תאימות בין גרסאות קפואות, וגם בין Top of Tree (ToT) לבין הגרסה הקפואה האחרונה.
בנוסף, צריך לנהל את הגדרת ה-API של גרסת ToT. בכל פעם שמתבצע עדכון של ממשק API, מריצים את הפקודה foo-update-api כדי לעדכן את הקובץ aidl_api/name/current
שמכיל את הגדרת ה-API של גרסת ToT.
כדי לשמור על יציבות הממשק, הבעלים יכולים להוסיף:
- שיטות בסוף ממשק (או שיטות עם סדרות חדשות שהוגדרו במפורש)
- אלמנטים לסוף של parcelable (צריך להוסיף ברירת מחדל לכל אלמנט)
- ערכים קבועים
- ב-Android 11, מונים
- ב-Android 12, שדות בסוף איחוד
אסור לבצע פעולות אחרות, ואף אחד אחר לא יכול לשנות ממשק (אחרת הוא עלול להיכנס לקונפליקט עם שינויים שהבעלים מבצע).
כדי לבדוק שכל הממשקים קפואים לקראת השקה, אפשר לבנות את הגרסה עם הגדרת משתני הסביבה הבאים:
AIDL_FROZEN_REL=true m ...
– ה-build מחייב הקפאה של כל ממשקי ה-AIDL היציבים שלא צוין להם שדהowner:
.AIDL_FROZEN_OWNERS="aosp test"
– כדי לבצע build, צריך להקפיא את כל ממשקי ה-AIDL היציבים עם השדהowner:
שמוגדר כ-"aosp" או כ-"test".
יציבות של ייבוא
עדכון הגרסאות של הייבוא לגרסאות קפואות של ממשק תואם לאחור בשכבת Stable AIDL. עם זאת, כדי לעדכן את הנתונים האלה צריך לעדכן את כל השרתים והלקוחות שמשתמשים בגרסה קודמת של הממשק, ויכול להיות שחלק מהאפליקציות יתבלבלו אם יתבצע שילוב בין גרסאות שונות של סוגים. באופן כללי, עבור חבילות של סוגים בלבד או חבילות נפוצות, זה בטוח כי כבר צריך לכתוב קוד לטיפול בסוגים לא מוכרים מעסקאות IPC.
בקוד של פלטפורמת Android, android.hardware.graphics.common
הוא הדוגמה הגדולה ביותר לשדרוג גרסה מהסוג הזה.
שימוש בממשקים עם גרסאות
שיטות ממשק
במהלך זמן הריצה, כשמנסים להפעיל שיטות חדשות בשרת ישן, לקוחות חדשים מקבלים שגיאה או חריגה, בהתאם לקצה העורפי.
- הקצה העורפי של
cpp
מקבל את::android::UNKNOWN_TRANSACTION
. - הקצה העורפי של
ndk
מקבל אתSTATUS_UNKNOWN_TRANSACTION
. - הקצה העורפי של
java
מקבל את הערךandroid.os.RemoteException
עם הודעה על כך שה-API לא מיושם.
לקבלת אסטרטגיות לטיפול בבעיה הזו, ראו שליחת שאילתות לגרסאות ושימוש בברירת המחדל.
Parcelables
כשמוסיפים שדות חדשים ל-Parcelable, לקוחות ושרתים ישנים משמיטים אותם. כששרתים ולקוחות חדשים מקבלים חבילות נתונים ישנות, ערכי ברירת המחדל של השדות החדשים מתמלאים באופן אוטומטי. כלומר, צריך לציין ברירת מחדל לכל השדות החדשים ב-Parcelable.
לקוחות לא צריכים לצפות ששרתים ישתמשו בשדות החדשים, אלא אם הם יודעים שהשרת מטמיע את הגרסה שבה השדה מוגדר (ראו שליחת שאילתות לגרסאות).
ערכים מוגדרים מראש וקבועים
באופן דומה, לקוחות ושרתים צריכים לדחות או להתעלם מערכים קבועים וממספרים מונים שלא מזוהים, לפי הצורך, כי יכול להיות שעוד יתווספו בעתיד. לדוגמה, שרת לא אמור לבטל את הפעולה כשמקבל מונה שהוא לא מכיר. השרת צריך להתעלם מהמונה או להחזיר משהו כדי שהלקוח יידע שאין תמיכה בו בהטמעה הזו.
איגודים
ניסיון לשלוח איחוד עם שדה חדש נכשל אם המקבל הוא ישן ולא יודע על השדה. השילוב עם השדה החדש אף פעם לא יופיע בהטמעה. המערכת מתעלמת מהכישלון אם מדובר בעסקה חד-כיוונית. אחרת, השגיאה היא BAD_VALUE
(לקצה העורפי של C++ או NDK) או IllegalArgumentException
(לקצה העורפי של Java). השגיאה מתקבלת אם הלקוח שולח שילוב שמוגדר לשדה החדש לשרת ישן, או אם מדובר בלקוח ישן שמקבל את השילוב משרת חדש.
ניהול כמה גרסאות
במרחב השמות של ה-linker ב-Android יכולה להיות רק גרסה אחת של ממשק aidl
ספציפי, כדי למנוע מצבים שבהם לסוגים של aidl
שנוצרים יש כמה הגדרות. ב-C++ יש את כלל ההגדרה היחידה, שמחייב הגדרה אחת בלבד של כל סמל.
ב-build של Android מופיעה שגיאה כשמודול תלוי בגרסאות שונות של אותה ספרייה aidl_interface
. יכול להיות שהמודול תלוי בספריות האלה באופן ישיר או עקיף, דרך יחסי התלות שלהן. השגיאות האלה מציגות את תרשים התלות מהמודול הפגום לגרסאות המתנגשות של ספריית aidl_interface
. צריך לעדכן את כל יחסי התלות כך שיכללו את אותה גרסה (בדרך כלל הגרסה האחרונה) של הספריות האלה.
אם הרבה מודולים שונים משתמשים בספריית הממשק, כדאי ליצור את cc_defaults
, java_defaults
ו-rust_defaults
לכל קבוצה של ספריות ותהליכים שצריכים להשתמש באותה גרסה. כשמשיקים גרסה חדשה של הממשק, אפשר לעדכן את הגדרות ברירת המחדל האלה וכל המודולים שמשתמשים בהן מתעדכנים יחד, כדי לוודא שהם לא משתמשים בגרסאות שונות של הממשק.
cc_defaults {
name: "my.aidl.my-process-group-ndk-shared",
shared_libs: ["my.aidl-V3-ndk"],
...
}
cc_library {
name: "foo",
defaults: ["my.aidl.my-process-group-ndk-shared"],
...
}
cc_binary {
name: "bar",
defaults: ["my.aidl.my-process-group-ndk-shared"],
...
}
כשמודולים של aidl_interface
מייבאים מודולים אחרים של aidl_interface
, נוצרות יחסי תלות נוספים שמחייבים שימוש בגרסאות ספציפיות יחד. המצב הזה יכול להיות קשה לניהול כשיש מודולים נפוצים של aidl_interface
שיובאו במספר מודולים של aidl_interface
שמשמשים יחד באותם תהליכים.
אפשר להשתמש ב-aidl_interfaces_defaults
כדי לשמור הגדרה אחת של הגרסאות העדכניות ביותר של יחסי התלות של aidl_interface
, שאפשר לעדכן במקום אחד, ושכל המודולים של aidl_interface
שרוצים לייבא את הממשק המשותף הזה יוכלו להשתמש בה.
aidl_interface_defaults {
name: "android.popular.common-latest-defaults",
imports: ["android.popular.common-V3"],
...
}
aidl_interface {
name: "android.foo",
defaults: ["my.aidl.latest-ndk-shared"],
...
}
aidl_interface {
name: "android.bar",
defaults: ["my.aidl.latest-ndk-shared"],
...
}
פיתוח מבוסס דגלים
אי אפשר להשתמש בממשקים שנמצאים בפיתוח (לא קפואים) במכשירים בגרסה היציבה, כי אין ערובה שהם יהיו תואמים לאחור.
AIDL תומך באפשרות חלופית בזמן ריצה לספריות הממשק הלא קפואות האלה, כדי שתוכלו לכתוב קוד לגרסה העדכנית ביותר שלא קופאה, ועדיין להשתמש בו במכשירי הגרסה הסופית. ההתנהגות של לקוחות עם תאימות לאחור דומה להתנהגות הקיימת, וגם ההטמעות צריכות לפעול בהתאם לאותה התנהגות. שימוש בממשקים עם גרסאות
דגל build של AIDL
הדגל RELEASE_AIDL_USE_UNFROZEN
שמגדיר את ההתנהגות הזו מוגדר ב-build/release/build_flags.bzl
. הערך true
מציין שהגרסה הלא קפואה של הממשק משמשת בזמן הריצה, והערך false
מציין שכל הספריות של הגרסאות הלא קפואות מתנהגות כמו הגרסה הקפואה האחרונה שלהן.
אפשר לשנות את הדגל ל-true
לפיתוח מקומי, אבל צריך להחזיר אותו ל-false
לפני הפרסום. בדרך כלל, הפיתוח מתבצע עם הגדרה שבה הדגל מוגדר ל-true
.
מטריצות ותאימות של מניפסט
אובייקטים של ממשק הספק (אובייקטי VINTF) מגדירים את הגרסאות הצפויות ואת הגרסאות שסופקו בכל צד של ממשק הספק.
רוב המכשירים שאינם Cuttlefish מטרגטים את מטריצת התאימות העדכנית ביותר רק אחרי שהממשקים קופאים, כך שאין הבדל בספריות AIDL שמבוססות על 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
כדי לשנות את קובצי המניפסט בזמן ה-build על סמך הערך של RELEASE_AIDL_USE_UNFROZEN
.
המניפסטים והקטעים של המניפסט מגדירים איזו גרסה של ממשק השירות מטמיע. כשמשתמשים בגרסה האחרונה של ממשק שלא הושהתה, צריך לעדכן את המניפסט כך שישקף את הגרסה החדשה. כשRELEASE_AIDL_USE_UNFROZEN=false
, רשומות המניפסט מותאמות על ידי libvintf
כדי לשקף את השינוי בספריית ה-AIDL שנוצרה. הגרסה משתנה מהגרסה שלא הושהתה, N
, לגרסה האחרונה שהושהתה, N - 1
. לכן, המשתמשים לא צריכים לנהל כמה מניפסטים או קטעי מניפסט לכל אחד מהשירותים שלהם.
שינויים בלקוח HAL
קוד הלקוח של HAL חייב להיות תואם לאחור לכל גרסה קודמת קפואה נתמכת. כשהערך של RELEASE_AIDL_USE_UNFROZEN
הוא false
, השירותים תמיד נראים כמו הגרסה האחרונה שהקפיאו או גרסה קודמת (לדוגמה, קריאה לשיטות חדשות שלא קפאו מחזירה את הערך UNKNOWN_TRANSACTION
, או לשדות parcelable
חדשים יש את ערכי ברירת המחדל שלהם). לקוחות של Android framework חייבים להיות תואמים לאחור לגרסאות קודמות נוספות, אבל זהו פרט חדש לגבי לקוחות של ספקים ולקוחות של ממשקים בבעלות שותפים.
שינויים בהטמעת HAL
ההבדל הגדול ביותר בפיתוח HAL לעומת פיתוח שמבוסס על דגלים הוא הדרישה שהטמעות HAL יהיו תואמות לאחור לגרסה הקפואה האחרונה, כדי לפעול כש-RELEASE_AIDL_USE_UNFROZEN
הוא false
.
התייחסות לתאימות לאחור בהטמעות ובקוד המכשיר היא תרגול חדש. שימוש בממשקים עם גרסאות
בדרך כלל, השיקולים לגבי תאימות לאחור זהים ללקוחות ולשרתים, ולקוד של המסגרת ולקוד של הספק, אבל יש הבדלים עדינים שחשוב לדעת עליהם, כי עכשיו אתם מטמיעים למעשה שתי גרסאות שמשתמשות באותו קוד מקור (הגרסה הנוכחית, שלא הושהתה).
דוגמה: לממשק יש שלוש גרסאות קפואות. הממשק מעודכן בשיטה חדשה. הלקוח והשירות מעודכנים לשימוש בספרייה החדשה בגרסה 4. מכיוון שספריית V4 מבוססת על גרסה לא קפואה של הממשק, היא מתנהגת כמו הגרסה הקפואה האחרונה, גרסה 3, כשRELEASE_AIDL_USE_UNFROZEN
הוא false
, ומונעת את השימוש בשיטה החדשה.
כשהממשק קפוא, כל הערכים של RELEASE_AIDL_USE_UNFROZEN
משתמשים בגרסה הקפואה הזו, וניתן להסיר את הקוד שמטפל בתאימות לאחור.
כשקוראים לשיטות בקריאות חזרה (callbacks), צריך לטפל בצורה תקינה במקרה שבו החזרה היא UNKNOWN_TRANSACTION
. יכול להיות שלקוחות יטמיעו שתי גרסאות שונות של קריאה חוזרת (callback) על סמך הגדרת המהדורה, לכן אי אפשר להניח שהלקוח שולח את הגרסה החדשה ביותר, וששיטות חדשות עשויות להחזיר את הערך הזה. המצב הזה דומה לאופן שבו לקוחות AIDL יציבים שומרים על תאימות לאחור עם שרתים, כפי שמתואר בקטע שימוש בממשקים עם גרסאות.
// 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
בקובץ ה-makefile של המכשיר:
ifeq ($(RELEASE_AIDL_USE_UNFROZEN),true)
PRODUCT_PACKAGES += \
android.hardware.health.storage-service
endif
אם השירות הוא חלק מתהליך גדול יותר, כך שלא ניתן להוסיף אותו למכשיר באופן מותנה, אפשר לבדוק אם השירות מוגדר באמצעות IServiceManager::isDeclared()
. אם הוא הוצהר ולא הצליח להירשם, צריך לבטל את התהליך. אם לא תצהירו עליו, סביר להניח שהוא לא ירשם.
Cuttlefish ככלי פיתוח
בכל שנה אחרי שה-VINTF מושעה, אנחנו משנים את מטריצת התאימות של המסגרת (FCM) target-level
ואת PRODUCT_SHIPPING_API_LEVEL
של Cuttlefish כך שישקפו את המכשירים שיושקו במהדורה של השנה הבאה. אנחנו משנים את הערכים של target-level
ו-PRODUCT_SHIPPING_API_LEVEL
כדי לוודא שיש מכשיר שיושקו שעבר בדיקה ועומדים בדרישות החדשות להשקה בשנה הבאה.
כשהערך של RELEASE_AIDL_USE_UNFROZEN
הוא true
, המערכת Cuttlefish משמשת לפיתוח גרסאות עתידיות של Android. היא מטרגטת את רמת ה-FCM של גרסת Android שתושק בשנה הבאה ואת PRODUCT_SHIPPING_API_LEVEL
, ולכן היא צריכה לעמוד בדרישות התוכנה של הספק (VSR) של הגרסה הבאה.
כשהערך של RELEASE_AIDL_USE_UNFROZEN
הוא false
, הערכים הקודמים של target-level
ו-PRODUCT_SHIPPING_API_LEVEL
ב-Cuttlefish משקפים מכשיר להשקה.
ב-Android 14 וגרסאות ישנות יותר, ההבחנה הזו תתבצע באמצעות הענפים השונים ב-Git שלא מביאים בחשבון את השינוי ב-FCM target-level
, ברמת ה-API למשלוח או בכל קוד אחר שמטרגט את הגרסה הבאה.
כללים למתן שמות למודולים
ב-Android 11, המערכת יוצרת באופן אוטומטי מודול של ספריית stub לכל שילוב של הגרסאות והקצוות העורפיים שמופעלים. כדי להפנות למודול ספציפי של ספריית stub לצורך קישור, לא משתמשים בשם המודול aidl_interface
, אלא בשם של מודול ספריית ה-stub, שהוא ifacename-version-backend, כאשר
ifacename
: שם המודולaidl_interface
version
הוא אחד מVversion-number
לגרסאות שהפסקנו לעדכןVlatest-frozen-version-number + 1
לגרסה של קוד העץ (עדיין לא הוקפאה)
backend
הוא אחד מjava
לקצה העורפי של Java,cpp
לקצה העורפי של C++,ndk
אוndk_platform
לקצה העורפי של NDK. הקוד הראשון מיועד לאפליקציות, והקוד השני מיועד לשימוש בפלטפורמה עד Android 13. ב-Android מגרסה 13 ואילך, צריך להשתמש רק ב-ndk
.rust
לקצה העורפי של Rust.
נניח שיש מודול בשם foo, שהגרסה העדכנית ביותר שלו היא 2, והוא תומך גם ב-NDK וגם ב-C++. במקרה כזה, AIDL יוצר את המודולים הבאים:
- על סמך גרסה 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
, שהתייחס לגרסה של ToT, הופך ל-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
שימו לב שמהדר ה-AIDL לא יוצר מודול בגרסה unstable
או מודול ללא גרסה לממשק AIDL יציב.
החל מ-Android 12, שם המודול שנוצר מממשק AIDL יציב תמיד כולל את הגרסה שלו.
שיטות חדשות לממשק המטא
ב-Android 10 נוספו כמה שיטות של ממשק מטא ל-AIDL היציב.
שליחת שאילתה לגרסה של הממשק של האובייקט המרוחק
לקוחות יכולים לשלוח שאילתה לגבי הגרסה והגיבוב של הממשק שהאובייקט המרוחק מטמיע, ולהשוות את הערכים המוחזרים לערכי הממשק שבו הלקוח משתמש.
דוגמה לקצה העורפי 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()
באופן הבא (super
משמש במקום IFoo
כדי למנוע שגיאות העתקה והדבקה. יכול להיות שתצטרכו את ההערה @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
וכו') משותפות בין הלקוח לשרת (לדוגמה, הכיתות יכולות להיות ב-classpath של האתחול). כשמשתפים כיתות, השרת מקושר גם לגרסה החדשה ביותר של הכיתות, גם אם הוא נוצר באמצעות גרסה ישנה יותר של הממשק. אם ממשק המטא הזה מוטמע בכיתה המשותפת, הוא תמיד מחזיר את הגרסה החדשה ביותר. עם זאת, כשמטמיעים את השיטה כפי שמתואר למעלה, מספר הגרסה של הממשק מוטמע בקוד של השרת (כי IFoo.VERSION
הוא static final int
שמוטמע בקוד כשמתבצעת הפניה אליו), ולכן השיטה יכולה להחזיר את הגרסה המדויקת שבה השרת נוצר.
עבודה עם ממשקים ישנים יותר
יכול להיות שגרסת הלקוח עודכנה לגרסה החדשה יותר של ממשק AIDL, אבל השרת משתמש בממשק AIDL הישן. במקרים כאלה, קריאה לשיטה בממשק ישן מחזירה את הערך UNKNOWN_TRANSACTION
.
כשמשתמשים ב-AIDL יציב, ללקוחות יש יותר שליטה. בצד הלקוח, אפשר להגדיר הטמעה שמוגדרת כברירת מחדל לממשק AIDL. קריאה ל-method בהטמעת ברירת המחדל מתבצעת רק אם ה-method לא מוטמע בצד המרוחק (כי הוא נוצר באמצעות גרסת ממשק ישנה יותר). מאחר שהגדרות ברירת המחדל מוגדרות באופן גלובלי, אין להשתמש בהן בהקשרים שעשויים להיות משותפים.
דוגמה ב-C++ ב-Android 13 ואילך:
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(...);
אין צורך לספק את הטמעת ברירת המחדל של כל השיטות בממשק AIDL. אין צורך לשנות את הגדרת ברירת המחדל של הכיתה impl
למתודות שאתם יודעים בוודאות שיוטמעו בצד המרוחק (כי אתם יודעים שהמכשיר המרוחק נוצר כשהמתודות היו בתיאור של ממשק ה-AIDL).
המרת AIDL קיים ל-AIDL מובנה או יציב
אם יש לכם ממשק AIDL קיים וקוד שמשתמש בו, תוכלו להשתמש בשלבים הבאים כדי להמיר את הממשק לממשק AIDL יציב.
מזהים את כל יחסי התלות של הממשק. לכל חבילת API שהממשק תלוי בה, צריך לקבוע אם החבילה מוגדרת ב-AIDL יציב. אם לא מגדירים את הפרמטר, צריך להמיר את החבילה.
ממירים את כל ה-Parcelables בממשק ל-Parcelables יציבים (קבצי הממשק עצמם יכולים להישאר ללא שינוי). כדי לעשות זאת, צריך להביע את המבנה שלהם ישירות בקובצי AIDL. צריך לכתוב מחדש את כיתות הניהול כדי להשתמש בסוגי הנתונים החדשים. אפשר לעשות זאת לפני שיוצרים חבילת
aidl_interface
(בהמשך).יוצרים חבילת
aidl_interface
(כמו שמתואר למעלה) שמכילה את שם המודול, יחסי התלות שלו וכל מידע אחר שדרוש לכם. כדי שהנתונים יהיו יציבים (ולא רק מובְנים), צריך גם להגדיר להם גרסאות. מידע נוסף זמין במאמר ממשקי ניהול גרסאות.