עדכון OTA למכשירים שאינם מכשירי A/B עם מחיצות דינמיות

‫Android 10 תומך במחיצות דינמיות, מערכת חלוקה למחיצות במרחב המשתמש שיכולה ליצור, לשנות את הגודל ולמחוק מחיצות במהלך עדכונים דרך האוויר (OTA).

בדף הזה מוסבר איך לקוחות OTA משנים את הגודל של מחיצות דינמיות במהלך עדכון במכשירים שאינם A/B.

במכשירים שהם לא A/B, העדכון של מחיצות דינמיות מתבצע באמצעות updater בתוך חבילת העדכון.

עדכון מכשירי ההשקה

הקטע הזה רלוונטי למכשירים מסוג non-A/B שמופעלים עם תמיכה במחיצות דינמיות. המכשירים האלה משודרגים מ-Android 10 לגרסאות מתקדמות יותר.

יצירת חבילות עדכון

חבילות עדכון OTA נוצרות על ידי הסקריפט ota_from_target_files, שנמצא בתיקייה build/make/tools/releasetools. כברירת מחדל, הסקריפט יוצר חבילה שמעדכנת את המחיצות system ו-vendor. אם יש מחיצות דינמיות נוספות, כמו product,‏ product_services או odm, צריך ליצור את העדכונים שלהן בקוד ספציפי למכשיר.

כדי ליצור עדכונים, במודול Python המורחב, מטמיעים את FullOTA_GetBlockDifferences() ואת IncrementalOTA_GetBlockDifferences(). שתי הפונקציות האלה מחזירות רשימה של אובייקטים מסוג BlockDifference, שכל אחד מהם מתאר את תיקון העדכון שיחול על מחיצה. אין לשנות באופן ידני את המחיצות שמוחזרות על ידי שתי הפונקציות האלה או לאמת אותן במקום אחר, למשל ב-*_InstallBegin() או ב-*_InstallEnd().

דוגמה לעדכון של דור:

# device/yoyodyne/tardis/releasetools.py

import os
from common import BlockDifference, EmptyImage, GetUserImage

# The joined list of user image partitions of source and target builds.
# - Items should be added to the list if new dynamic partitions are added.
# - Items should not be removed from the list even if dynamic partitions are
#   deleted. When generating an incremental OTA package, this script needs to
#   know that an image is present in source build but not in target build.
USERIMAGE_PARTITIONS = [
    "product",
    "odm",
]

def GetUserImages(input_tmp, input_zip):
  return {partition: GetUserImage(partition, input_tmp, input_zip)
          for partition in USERIMAGE_PARTITIONS
          if os.path.exists(os.path.join(input_tmp,
                                         "IMAGES", partition + ".img"))}

def FullOTA_GetBlockDifferences(info):
  images = GetUserImages(info.input_tmp, info.input_zip)
  return [BlockDifference(partition, image)
          for partition, image in images.items()]

def IncrementalOTA_GetBlockDifferences(info):
  source_images = GetUserImages(info.source_tmp, info.source_zip)
  target_images = GetUserImages(info.target_tmp, info.target_zip)

  # Use EmptyImage() as a placeholder for partitions that will be deleted.
  for partition in source_images:
    target_images.setdefault(partition, EmptyImage())

  # Use source_images.get() because new partitions are not in source_images.
  return [BlockDifference(partition, target_image, source_images.get(partition))
          for partition, target_image in target_images.items()]

עדכון תהליך העבודה

מאחורי הקלעים, הפונקציות הבאות מתווספות לסקריפט edify:

  • unmap_partition(name)
    • אם המחיצה ממופה, מבטלים את המיפוי שלה. אחרת, לא מבצעים שום פעולה.
    • מחזירה את המחרוזת t אם הפעולה הצליחה, או מחרוזת ריקה אם הפעולה נכשלה.
  • map_partition(name)
    • ממפים את החלוקה אם היא עדיין לא ממופה.
    • מחזירה את הנתיב המוחלט של התקן הבלוקים הממופה במקרה של הצלחה, או מחרוזת ריקה במקרה של כשל.
  • update_dynamic_partitions(op_list)
    • החלת רשימת הפעולות הנתונה על מטא-נתונים של מחיצות דינמיות, ביטול המיפוי של מחיצות אם יש צורך בכך.
    • מחזירה t אם הפעולה הצליחה, או מחרוזת ריקה אם היא נכשלה.

הארגומנט op_list של update_dynamic_partitions מצביע על קובץ בחבילת העדכון. כל שורה בקובץ מציינת פעולה. אם פעולה כלשהי נכשלת, הפונקציה update_dynamic_partitions מחזירה מייד מחרוזת ריקה. הפעולות הן:

  • resize partition-name size
    • מבטלים את המיפוי של המחיצה, ואז משנים את הגודל שלה ל-size.
  • remove partition_name
    • מבטלים את המיפוי של המחיצה ואז מסירים אותה.
  • add partition-name group-name
    • הוספת מחיצה חדשה לקבוצה שצוינה.
    • הפעולה תבוטל אם הקבוצה לא קיימת או אם המחיצה כבר קיימת.
  • move partition-name group-name
    • העברת המחיצה לקבוצה שצוינה.
    • הפעולה תבוטל אם הקבוצה או החלוקה לא קיימות.
  • add_group group-name maximum-size
    • הוספת קבוצה עם השם שצוין והגודל המקסימלי.
    • הפעולה תבוטל אם הקבוצה כבר קיימת.
    • הערך maximum_size‏ 0 מציין שאין מגבלות גודל על המחיצות בקבוצה. נדרש בדיקה נוספת כדי לוודא שהמחיצות בקבוצה לא חורגות מהשטח הפנוי במכשיר.
  • resize_group group-name maximum-size
    • משנים את גודל הקבוצה לגודל המקסימלי שצוין.
    • הפעולה תבוטל אם הקבוצה לא קיימת.
    • הערך maximum_size‏ 0 מציין שאין מגבלות גודל על המחיצות בקבוצה. נדרש בדיקה נוספת כדי לוודא שהמחיצות בקבוצה לא חורגות מהמקום שזמין במכשיר.
  • remove_group group-name
    • להסיר קבוצה.
    • הפעולה תבוטל אם יש מחיצות בקבוצה.
  • remove_all_groups
    • ביטול המיפוי של כל המחיצות מ-Device Mapper.
    • מסירים את כל המחיצות והקבוצות.

עדכון OTA מצטבר

עדכוני OTA מצטברים פועלים לפי הלוגיקה הבאה:

  1. הקטנת מחיצות/מחיקת מחיצות/העברת מחיצות מחוץ לקבוצה (כדי שיהיה מספיק מקום להקטנת קבוצות)
  2. מצמצמים את הקבוצות (כדי שיהיה מספיק מקום להגדלת הקבוצות)
  3. הגדלת הקבוצות (כדי שיהיה לנו מספיק מקום להגדלה או להוספה של מחיצות)
  4. הגדלת המחיצות/הוספת מחיצות/העברת מחיצות לקבוצה חדשה

ההסבר המפורט על יצירת update-script:

for each shrinking partition:
    block_image_update(map_partition(name), …)

update_dynamic_partitions(op_list)

for each growing / adding partition:
    block_image_update(map_partition(name), …)

הקובץ op_list של update_dynamic_partitions נוצר לפי הלוגיקה הזו:

for each deleting partition:
    remove
for each partition that changes groups:
    move to "default"
for each shrinking partition:
    resize
for each shrinking / removing group:
    resize_group / remove_group
for each growing / adding group:
    resize_group / add_group
for each adding partition:
    add
for each growing / adding partition:
    resize
for each partition that changes groups:
    move to target group

‫Full OTA

עדכוני OTA מלאים מתבצעים לפי הלוגיקה הבאה:

  1. מחיקה של כל המחיצות והקבוצות הקיימות
  2. הוספת קבוצות
  3. הוספת מחיצות

ההסבר המפורט על יצירת update-script:

update_dynamic_partitions(op_list)

for each adding partition:
    block_image_update(map_partition(name), …)

הקובץ op_list של update_dynamic_partitions נוצר לפי הלוגיקה הזו:

remove_all_groups
for each adding group:
    add_group
for each adding partition:
    add
for each adding partition:
    resize