लेयर और डिसप्ले

लेयर और डिसप्ले, दो प्रिमिटिव हैं. ये कंपोज़िशन और डिसप्ले हार्डवेयर के साथ इंटरैक्शन को दिखाते हैं.

परतें

लेयर, कंपोज़िशन की सबसे अहम यूनिट होती है. लेयर, सरफ़ेस और SurfaceControl के इंस्टेंस का कॉम्बिनेशन होती है. हर लेयर में प्रॉपर्टी का एक सेट होता है. इससे यह तय होता है कि लेयर, दूसरी लेयर के साथ कैसे इंटरैक्ट करेगी. लेयर की प्रॉपर्टी के बारे में यहां दी गई टेबल में बताया गया है:

प्रॉपर्टी ब्यौरा
पोज़िशनल इससे यह तय होता है कि लेयर, डिसप्ले पर कहां दिखेगी. इसमें लेयर की सीमाओं की पोज़िशन और अन्य लेयर के मुकाबले उसका Z क्रम जैसी जानकारी शामिल होती है. इससे यह तय होता है कि लेयर को अन्य लेयर के ऊपर दिखाना है या नीचे.
कॉन्टेंट इससे पता चलता है कि लेयर पर मौजूद कॉन्टेंट, पोज़िशन से जुड़ी प्रॉपर्टी के सेट किए गए दायरे में कैसे दिखता है. इसमें काटना (कॉन्टेंट को लेयर की सीमाओं तक फैलाने के लिए) और बदलना (घुमाया गया या फ़्लिप किया गया कॉन्टेंट दिखाने के लिए) शामिल है.
कंपोज़िशन इससे यह तय किया जाता है कि लेयर को दूसरी लेयर के साथ कैसे कंपोज़ किया जाना चाहिए. इसमें अल्फ़ा कंपोज़िटिंग के लिए, ब्लेंडिंग मोड और लेयर-वाइड ऐल्फ़ा वैल्यू जैसी जानकारी शामिल होती है.
ऑप्टिमाइज़ेशन यह जानकारी, लेयर कंपोज़िशन को सही तरीके से बनाने के लिए ज़रूरी नहीं है. हालांकि, हार्डवेयर कंपोज़र (एचडब्ल्यूसी) डिवाइस इसका इस्तेमाल, कंपोज़िशन की परफ़ॉर्मेंस को ऑप्टिमाइज़ करने के लिए कर सकता है. इसमें लेयर का दिखने वाला हिस्सा और वह हिस्सा शामिल होता है जिसे पिछले फ़्रेम के बाद अपडेट किया गया है.

डिसप्ले

डिसप्ले, कंपोज़िशन की एक और अहम यूनिट है. किसी सिस्टम में एक से ज़्यादा डिसप्ले हो सकते हैं. साथ ही, सिस्टम के सामान्य ऑपरेशन के दौरान डिसप्ले जोड़े या हटाए जा सकते हैं. एचडब्ल्यूसी के अनुरोध पर या फ़्रेमवर्क के अनुरोध पर डिसप्ले जोड़े या हटाए जाते हैं.

जब किसी डिवाइस से कोई बाहरी डिसप्ले कनेक्ट या डिसकनेक्ट किया जाता है, तो HWC डिवाइस, डिसप्ले को जोड़ने या हटाने का अनुरोध करता है. इसे हॉटप्लगिंग कहा जाता है. क्लाइंट, वर्चुअल डिसप्ले का अनुरोध करते हैं. इनके कॉन्टेंट को फ़िज़िकल डिसप्ले के बजाय, ऑफ़-स्क्रीन बफ़र में रेंडर किया जाता है.

वर्चुअल डिसप्ले

SurfaceFlinger इनके साथ काम करता है: फ़ोन या टैबलेट में बना इंटरनल डिसप्ले, एचडीएमआई से कनेक्ट किए गए टेलीविज़न जैसे एक्सटर्नल डिसप्ले, और एक या उससे ज़्यादा वर्चुअल डिसप्ले. ये वर्चुअल डिसप्ले, सिस्टम में कंपोज़िट आउटपुट उपलब्ध कराते हैं. वर्चुअल डिसप्ले का इस्तेमाल, स्क्रीन रिकॉर्ड करने या स्क्रीन को नेटवर्क पर भेजने के लिए किया जा सकता है. वर्चुअल डिसप्ले के लिए जनरेट किए गए फ़्रेम, BufferQueue में लिखे जाते हैं.

वर्चुअल डिसप्ले, मुख्य डिसप्ले (लेयर स्टैक) के साथ लेयर का एक ही सेट शेयर कर सकते हैं या उनका अपना सेट हो सकता है. वर्चुअल डिसप्ले के लिए कोई VSync नहीं है. इसलिए, इंटरनल डिसप्ले के लिए VSync, सभी डिसप्ले के लिए कंपोज़िशन को ट्रिगर करता है.

एचडब्ल्यूसी के ऐसे वर्शन जिन पर ये सुविधाएं काम करती हैं उनमें वर्चुअल डिसप्ले को OpenGL ES (GLES), एचडब्ल्यूसी या GLES और एचडब्ल्यूसी, दोनों के साथ कंपोज़ किया जा सकता है. जिन डिवाइसों पर वर्चुअल डिसप्ले की सुविधा काम नहीं करती है उन पर वर्चुअल डिसप्ले हमेशा GLES का इस्तेमाल करके कंपोज़ किए जाते हैं.

केस स्टडी: screenrecord

screenrecord कमांड की मदद से उपयोगकर्ता, स्क्रीन पर दिखने वाली हर चीज़ को डिस्क पर MP4 फ़ाइल के तौर पर रिकॉर्ड कर सकता है. इसे लागू करने के लिए, सिस्टम को SurfaceFlinger से कंपोज़ किए गए फ़्रेम मिलते हैं. इसके बाद, सिस्टम उन्हें वीडियो एन्कोडर में लिखता है. इसके बाद, एन्कोड किए गए वीडियो डेटा को किसी फ़ाइल में लिखता है. वीडियो कोडेक को एक अलग प्रोसेस (mediaserver) मैनेज करती है. इसलिए, बड़े ग्राफ़िक बफ़र को सिस्टम में इधर-उधर ले जाना पड़ता है. इसे और भी मुश्किल बनाने के लिए, फ़ुल रिज़ॉल्यूशन पर 60 एफ़पीएस वाला वीडियो रिकॉर्ड करना है. BufferQueue की मदद से, इस सुविधा को बेहतर तरीके से काम करने में मदद मिलती है.

MediaCodec क्लास की मदद से, कोई ऐप्लिकेशन डेटा को बफ़र में रॉ बाइट के तौर पर या किसी सरफेस के ज़रिए उपलब्ध करा सकता है. जब screenrecord किसी वीडियो एनकोडर के ऐक्सेस का अनुरोध करता है, तो mediaserver प्रोसेस एक BufferQueue बनाती है. यह खुद को उपभोक्ता पक्ष से कनेक्ट करती है. इसके बाद, यह प्रोड्यूसर साइड को screenrecord को एक सरफेस के तौर पर वापस भेजती है.

इसके बाद, screenrecord यूटिलिटी, SurfaceFlinger से एक वर्चुअल डिसप्ले बनाने के लिए कहती है. यह वर्चुअल डिसप्ले, मुख्य डिसप्ले की तरह ही काम करता है. इसका मतलब है कि इसमें मुख्य डिसप्ले की सभी लेयर होती हैं. साथ ही, यह वर्चुअल डिसप्ले को mediaserver प्रोसेस से मिले सर्फ़ेस पर आउटपुट भेजने का निर्देश देती है. इस मामले में, SurfaceFlinger, बफ़र का इस्तेमाल करने के बजाय उन्हें बनाता है.

कॉन्फ़िगरेशन पूरा होने के बाद, कोड में बदला गया डेटा दिखने पर screenrecord ट्रिगर होता है. ऐप्लिकेशन के ड्रॉ करने पर, उनके बफ़र SurfaceFlinger पर जाते हैं. SurfaceFlinger, इन बफ़र को एक बफ़र में कंपोज़ करता है. इसके बाद, इसे सीधे mediaserver प्रोसेस में मौजूद वीडियो एनकोडर को भेज दिया जाता है. screenrecord प्रोसेस में पूरे फ़्रेम कभी नहीं दिखते. आंतरिक तौर पर, mediaserver प्रोसेस में बफ़र को इधर-उधर ले जाने का अपना तरीका होता है. यह हैंडल के ज़रिए भी डेटा पास करता है, जिससे ओवरहेड कम हो जाता है.

केस स्टडी: अतिरिक्त वर्चुअल डिसप्ले बनाने की सेटिंग

WindowManager, SurfaceFlinger से ऐसी लेयर बनाने के लिए कह सकता है जो दिखती हो. इसके लिए, SurfaceFlinger, BufferQueue उपभोक्ता के तौर पर काम करता है. SurfaceFlinger से वर्चुअल डिसप्ले बनाने के लिए भी कहा जा सकता है. इसके लिए, SurfaceFlinger, BufferQueue प्रोड्यूसर के तौर पर काम करता है.

अगर किसी वर्चुअल डिसप्ले को दिखने वाली लेयर से कनेक्ट किया जाता है, तो एक क्लोज़्ड लूप बनता है. इसमें कंपोज़िट की गई स्क्रीन, विंडो में दिखती है. वह विंडो अब कंपोज़िट किए गए आउटपुट का हिस्सा है. इसलिए, अगली बार रीफ़्रेश करने पर, विंडो के अंदर कंपोज़िट की गई इमेज में विंडो का कॉन्टेंट भी दिखेगा. इसे आज़माने के लिए, सेटिंग में जाकर डेवलपर के लिए सेटिंग और टूल चालू करें. इसके बाद, सेकंडरी डिसप्ले सिम्युलेट करें को चुनें और कोई विंडो चालू करें. सेकंडरी डिसप्ले को ऐक्शन में देखने के लिए, screenrecord का इस्तेमाल करके डिसप्ले चालू करने की प्रोसेस को रिकॉर्ड करें. इसके बाद, इसे एक-एक फ़्रेम करके देखें.