cmsdk: Create Concierge parcel handler concept.

Simply, Concierge handles your parcels and makes sure they
  get marshalled and unmarshalled correctly when cross IPC
  boundaries even when there is a version mismatch between the client
  sdk level and the framework implementation.

 On incoming parcel (to be unmarshalled):
     ParcelInfo incomingParcelInfo = Concierge.receiveParcel(incomingParcel);
     int parcelableVersion = incomingParcelInfo.getParcelVersion();

     // Do unmarshalling steps here iterating over every plausible version

     // Complete the process
     incomingParcelInfo.complete();

 On outgoing parcel (to be marshalled):
     ParcelInfo outgoingParcelInfo = Concierge.prepareParcel(incomingParcel);

     // Do marshalling steps here iterating over every plausible version

     // Complete the process
     outgoingParcelInfo.complete();

Change-Id: I0096851e08ce25908e19e39e6bc4af344e85725e
This commit is contained in:
Adnan Begovic 2016-03-29 11:41:22 -07:00
parent 4195a1cf89
commit eafcdbd229
21 changed files with 428 additions and 429 deletions

View File

@ -28,6 +28,9 @@ import android.util.Log;
import android.widget.RemoteViews; import android.widget.RemoteViews;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
import java.util.ArrayList; import java.util.ArrayList;
/** /**
@ -128,11 +131,9 @@ public class CustomTile implements Parcelable {
* Unflatten the CustomTile from a parcel. * Unflatten the CustomTile from a parcel.
*/ */
public CustomTile(Parcel parcel) { public CustomTile(Parcel parcel) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(parcel);
int parcelableVersion = parcel.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = parcel.readInt();
int startPosition = parcel.dataPosition();
// Pattern here is that all new members should be added to the end of // Pattern here is that all new members should be added to the end of
// the writeToParcel method. Then we step through each version, until the latest // the writeToParcel method. Then we step through each version, until the latest
@ -177,7 +178,8 @@ public class CustomTile implements Parcelable {
} }
} }
parcel.setDataPosition(startPosition + parcelableSize); // Complete parcel info for the concierge
parcelInfo.complete();
} }
/** /**
@ -267,15 +269,8 @@ public class CustomTile implements Parcelable {
@Override @Override
public void writeToParcel(Parcel out, int flags) { public void writeToParcel(Parcel out, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(out);
out.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = out.dataPosition();
out.writeInt(0);
int startPosition = out.dataPosition();
// ==== APRICOT ===== // ==== APRICOT =====
if (onClick != null) { if (onClick != null) {
@ -341,11 +336,8 @@ public class CustomTile implements Parcelable {
out.writeInt(0); out.writeInt(0);
} }
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = out.dataPosition() - startPosition; parcelInfo.complete();
out.setDataPosition(sizePosition);
out.writeInt(parcelableSize);
out.setDataPosition(startPosition + parcelableSize);
} }
/** /**
@ -382,11 +374,9 @@ public class CustomTile implements Parcelable {
private int styleId; private int styleId;
private ExpandedStyle(Parcel parcel) { private ExpandedStyle(Parcel parcel) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(parcel);
int parcelableVersion = parcel.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = parcel.readInt();
int startPosition = parcel.dataPosition();
// Pattern here is that all new members should be added to the end of // Pattern here is that all new members should be added to the end of
// the writeToParcel method. Then we step through each version, until the latest // the writeToParcel method. Then we step through each version, until the latest
@ -404,7 +394,8 @@ public class CustomTile implements Parcelable {
} }
} }
parcel.setDataPosition(startPosition + parcelableSize); // Complete parcel info for the concierge
parcelInfo.complete();
} }
/** /**
@ -473,15 +464,8 @@ public class CustomTile implements Parcelable {
@Override @Override
public void writeToParcel(Parcel parcel, int i) { public void writeToParcel(Parcel parcel, int i) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(parcel);
parcel.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = parcel.dataPosition();
parcel.writeInt(0);
int startPosition = parcel.dataPosition();
// ==== APRICOT ==== // ==== APRICOT ====
if (expandedItems != null) { if (expandedItems != null) {
@ -500,11 +484,8 @@ public class CustomTile implements Parcelable {
parcel.writeInt(0); parcel.writeInt(0);
} }
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = parcel.dataPosition() - startPosition; parcelInfo.complete();
parcel.setDataPosition(sizePosition);
parcel.writeInt(parcelableSize);
parcel.setDataPosition(startPosition + parcelableSize);
} }
@Override @Override
@ -679,11 +660,9 @@ public class CustomTile implements Parcelable {
* Unflatten the ExpandedItem from a parcel. * Unflatten the ExpandedItem from a parcel.
*/ */
protected ExpandedItem(Parcel parcel) { protected ExpandedItem(Parcel parcel) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(parcel);
int parcelableVersion = parcel.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = parcel.readInt();
int startPosition = parcel.dataPosition();
// Pattern here is that all new members should be added to the end of // Pattern here is that all new members should be added to the end of
// the writeToParcel method. Then we step through each version, until the latest // the writeToParcel method. Then we step through each version, until the latest
@ -707,7 +686,8 @@ public class CustomTile implements Parcelable {
} }
} }
parcel.setDataPosition(startPosition + parcelableSize); // Complete parcel info for the concierge
parcelInfo.complete();
} }
@Override @Override
@ -717,15 +697,8 @@ public class CustomTile implements Parcelable {
@Override @Override
public void writeToParcel(Parcel out, int flags) { public void writeToParcel(Parcel out, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(out);
out.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = out.dataPosition();
out.writeInt(0);
int startPosition = out.dataPosition();
// ==== APRICOT ==== // ==== APRICOT ====
if (onClickPendingIntent != null) { if (onClickPendingIntent != null) {
@ -756,11 +729,8 @@ public class CustomTile implements Parcelable {
out.writeInt(0); out.writeInt(0);
} }
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = out.dataPosition() - startPosition; parcelInfo.complete();
out.setDataPosition(sizePosition);
out.writeInt(parcelableSize);
out.setDataPosition(startPosition + parcelableSize);
} }
@Override @Override

View File

@ -22,7 +22,10 @@ import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import android.text.TextUtils; import android.text.TextUtils;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
/** /**
* Data structure defining a Live lock screen. * Data structure defining a Live lock screen.
@ -86,11 +89,9 @@ public class LiveLockScreenInfo implements Parcelable {
} }
private LiveLockScreenInfo(Parcel source) { private LiveLockScreenInfo(Parcel source) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(source);
int version = source.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int size = source.readInt();
int start = source.dataPosition();
this.priority = source.readInt(); this.priority = source.readInt();
String component = source.readString(); String component = source.readString();
@ -98,7 +99,8 @@ public class LiveLockScreenInfo implements Parcelable {
? ComponentName.unflattenFromString(component) ? ComponentName.unflattenFromString(component)
: null; : null;
source.setDataPosition(start + size); // Complete parcel info for the concierge
parcelInfo.complete();
} }
@Override @Override
@ -108,23 +110,14 @@ public class LiveLockScreenInfo implements Parcelable {
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
dest.writeInt(Build.PARCELABLE_VERSION);
int sizePos = dest.dataPosition();
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
dest.writeInt(0);
int dataStartPos = dest.dataPosition();
dest.writeInt(priority); dest.writeInt(priority);
dest.writeString(component != null ? component.flattenToString() : ""); dest.writeString(component != null ? component.flattenToString() : "");
// Go back and write size // Complete the parcel info for the concierge
int size = dest.dataPosition() - dataStartPos; parcelInfo.complete();
dest.setDataPosition(sizePos);
dest.writeInt(size);
dest.setDataPosition(dataStartPos + size);
} }
@Override @Override

View File

@ -35,13 +35,15 @@ import cyanogenmod.profiles.LockSettings;
import cyanogenmod.profiles.RingModeSettings; import cyanogenmod.profiles.RingModeSettings;
import cyanogenmod.profiles.StreamSettings; import cyanogenmod.profiles.StreamSettings;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
import org.xmlpull.v1.XmlPullParser; import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException; import org.xmlpull.v1.XmlPullParserException;
import java.io.IOException; import java.io.IOException;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collection; import java.util.Collection;
import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
@ -227,11 +229,9 @@ public final class Profile implements Parcelable, Comparable {
} }
private ProfileTrigger(Parcel in) { private ProfileTrigger(Parcel in) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(in);
int parcelableVersion = in.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = in.readInt();
int startPosition = in.dataPosition();
// Pattern here is that all new members should be added to the end of // Pattern here is that all new members should be added to the end of
// the writeToParcel method. Then we step through each version, until the latest // the writeToParcel method. Then we step through each version, until the latest
@ -243,31 +243,22 @@ public final class Profile implements Parcelable, Comparable {
mName = in.readString(); mName = in.readString();
} }
in.setDataPosition(startPosition + parcelableSize); // Complete parcel info for the concierge
parcelInfo.complete();
} }
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
dest.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
dest.writeInt(mType); dest.writeInt(mType);
dest.writeString(mId); dest.writeString(mId);
dest.writeInt(mState); dest.writeInt(mState);
dest.writeString(mName); dest.writeString(mName);
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = dest.dataPosition() - startPosition; parcelInfo.complete();
dest.setDataPosition(sizePosition);
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
@Override @Override
@ -547,15 +538,8 @@ public final class Profile implements Parcelable, Comparable {
/** @hide */ /** @hide */
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
dest.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
// === BOYSENBERRY === // === BOYSENBERRY ===
if (!TextUtils.isEmpty(mName)) { if (!TextUtils.isEmpty(mName)) {
@ -649,21 +633,15 @@ public final class Profile implements Parcelable, Comparable {
dest.writeInt(0); dest.writeInt(0);
} }
// Complete the parcel info for the concierge
// Go back and write size parcelInfo.complete();
int parcelableSize = dest.dataPosition() - startPosition;
dest.setDataPosition(sizePosition);
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
/** @hide */ /** @hide */
public void readFromParcel(Parcel in) { public void readFromParcel(Parcel in) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(in);
int parcelableVersion = in.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = in.readInt();
int startPosition = in.dataPosition();
// Pattern here is that all new members should be added to the end of // Pattern here is that all new members should be added to the end of
// the writeToParcel method. Then we step through each version, until the latest // the writeToParcel method. Then we step through each version, until the latest
@ -735,7 +713,8 @@ public final class Profile implements Parcelable, Comparable {
} }
} }
in.setDataPosition(startPosition + parcelableSize); // Complete the parcel info for the concierge
parcelInfo.complete();
} }
/** /**

View File

@ -18,9 +18,6 @@ package cyanogenmod.app;
import android.app.Notification; import android.app.Notification;
import android.app.NotificationGroup; import android.app.NotificationGroup;
import cyanogenmod.os.Build;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException;
import android.content.Context; import android.content.Context;
import android.database.Cursor; import android.database.Cursor;
@ -32,9 +29,17 @@ import android.os.ParcelUuid;
import android.text.TextUtils; import android.text.TextUtils;
import android.util.Log; import android.util.Log;
import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
import java.io.IOException; import java.io.IOException;
import java.util.UUID; import java.util.UUID;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException;
/** /**
* @hide * @hide
* TODO: This isn't ready for public use * TODO: This isn't ready for public use
@ -264,15 +269,8 @@ public final class ProfileGroup implements Parcelable {
/** @hide */ /** @hide */
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
dest.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
// === BOYSENBERRY === // === BOYSENBERRY ===
dest.writeString(mName); dest.writeString(mName);
@ -286,20 +284,15 @@ public final class ProfileGroup implements Parcelable {
dest.writeString(mVibrateMode.name()); dest.writeString(mVibrateMode.name());
dest.writeString(mLightsMode.name()); dest.writeString(mLightsMode.name());
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = dest.dataPosition() - startPosition; parcelInfo.complete();
dest.setDataPosition(sizePosition);
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
/** @hide */ /** @hide */
public void readFromParcel(Parcel in) { public void readFromParcel(Parcel in) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(in);
int parcelableVersion = in.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = in.readInt();
int startPosition = in.dataPosition();
// Pattern here is that all new members should be added to the end of // Pattern here is that all new members should be added to the end of
// the writeToParcel method. Then we step through each version, until the latest // the writeToParcel method. Then we step through each version, until the latest
@ -318,7 +311,8 @@ public final class ProfileGroup implements Parcelable {
mLightsMode = Mode.valueOf(Mode.class, in.readString()); mLightsMode = Mode.valueOf(Mode.class, in.readString());
} }
in.setDataPosition(startPosition + parcelableSize); // Complete parcel info for the concierge
parcelInfo.complete();
} }
public enum Mode { public enum Mode {

View File

@ -19,8 +19,12 @@ package cyanogenmod.app;
import android.os.Parcel; import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import android.os.UserHandle; import android.os.UserHandle;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
/** /**
* Class encapsulating a Custom Tile. Sent by the StatusBarManagerService to clients including * Class encapsulating a Custom Tile. Sent by the StatusBarManagerService to clients including
* the status bar panel and any {@link cyanogenmod.app.CustomTileListenerService} clients. * the status bar panel and any {@link cyanogenmod.app.CustomTileListenerService} clients.
@ -67,11 +71,9 @@ public class StatusBarPanelCustomTile implements Parcelable {
public StatusBarPanelCustomTile(Parcel in) { public StatusBarPanelCustomTile(Parcel in) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(in);
int parcelableVersion = in.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = in.readInt();
int startPosition = in.dataPosition();
// tmp variables for final // tmp variables for final
String tmpResPkg = null; String tmpResPkg = null;
@ -122,7 +124,8 @@ public class StatusBarPanelCustomTile implements Parcelable {
this.postTime = tmpPostTime; this.postTime = tmpPostTime;
this.key = key(); this.key = key();
in.setDataPosition(startPosition + parcelableSize); // Complete parcel info for the concierge
parcelInfo.complete();
} }
private String key() { private String key() {
@ -157,15 +160,8 @@ public class StatusBarPanelCustomTile implements Parcelable {
@Override @Override
public void writeToParcel(Parcel out, int flags) { public void writeToParcel(Parcel out, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(out);
out.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = out.dataPosition();
out.writeInt(0);
int startPosition = out.dataPosition();
// ==== APRICOT === // ==== APRICOT ===
out.writeString(this.pkg); out.writeString(this.pkg);
@ -186,11 +182,8 @@ public class StatusBarPanelCustomTile implements Parcelable {
// ==== BOYSENBERRY ===== // ==== BOYSENBERRY =====
out.writeString(this.resPkg); out.writeString(this.resPkg);
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = out.dataPosition() - startPosition; parcelInfo.complete();
out.setDataPosition(sizePosition);
out.writeInt(parcelableSize);
out.setDataPosition(startPosition + parcelableSize);
} }
@Override @Override

View File

@ -20,7 +20,10 @@ import android.annotation.NonNull;
import android.net.Uri; import android.net.Uri;
import android.os.Parcel; import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
/** /**
* @hide * @hide
@ -55,11 +58,9 @@ public class ApplicationSuggestion implements Parcelable {
} }
private ApplicationSuggestion(Parcel in) { private ApplicationSuggestion(Parcel in) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(in);
int parcelableVersion = in.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = in.readInt();
int startPosition = in.dataPosition();
if (parcelableVersion >= Build.CM_VERSION_CODES.APRICOT) { if (parcelableVersion >= Build.CM_VERSION_CODES.APRICOT) {
mName = in.readString(); mName = in.readString();
@ -68,7 +69,8 @@ public class ApplicationSuggestion implements Parcelable {
mThumbnailUri = in.readParcelable(Uri.class.getClassLoader()); mThumbnailUri = in.readParcelable(Uri.class.getClassLoader());
} }
in.setDataPosition(startPosition + parcelableSize); // Complete parcel info for the concierge
parcelInfo.complete();
} }
@Override @Override
@ -78,26 +80,16 @@ public class ApplicationSuggestion implements Parcelable {
@Override @Override
public void writeToParcel(Parcel out, int flags) { public void writeToParcel(Parcel out, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(out);
out.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = out.dataPosition();
out.writeInt(0);
int startPosition = out.dataPosition();
out.writeString(mName); out.writeString(mName);
out.writeString(mPackage); out.writeString(mPackage);
out.writeParcelable(mDownloadUri, flags); out.writeParcelable(mDownloadUri, flags);
out.writeParcelable(mThumbnailUri, flags); out.writeParcelable(mThumbnailUri, flags);
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = out.dataPosition() - startPosition; parcelInfo.complete();
out.setDataPosition(sizePosition);
out.writeInt(parcelableSize);
out.setDataPosition(startPosition + parcelableSize);
} }
public String getName() { public String getName() {

View File

@ -20,6 +20,8 @@ import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
/** /**
* Display Modes API * Display Modes API
@ -45,11 +47,9 @@ public class DisplayMode implements Parcelable {
} }
private DisplayMode(Parcel parcel) { private DisplayMode(Parcel parcel) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(parcel);
int parcelableVersion = parcel.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = parcel.readInt();
int startPosition = parcel.dataPosition();
// temp vars // temp vars
int tmpId = -1; int tmpId = -1;
@ -65,7 +65,9 @@ public class DisplayMode implements Parcelable {
// set temps // set temps
this.id = tmpId; this.id = tmpId;
this.name = tmpName; this.name = tmpName;
parcel.setDataPosition(startPosition + parcelableSize);
// Complete parcel info for the concierge
parcelInfo.complete();
} }
@Override @Override
@ -75,15 +77,8 @@ public class DisplayMode implements Parcelable {
@Override @Override
public void writeToParcel(Parcel out, int flags) { public void writeToParcel(Parcel out, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(out);
out.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = out.dataPosition();
out.writeInt(0);
int startPosition = out.dataPosition();
// ==== BOYSENBERRY ===== // ==== BOYSENBERRY =====
out.writeInt(id); out.writeInt(id);
@ -94,11 +89,8 @@ public class DisplayMode implements Parcelable {
out.writeInt(0); out.writeInt(0);
} }
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = out.dataPosition() - startPosition; parcelInfo.complete();
out.setDataPosition(sizePosition);
out.writeInt(parcelableSize);
out.setDataPosition(startPosition + parcelableSize);
} }
/** @hide */ /** @hide */

View File

@ -27,17 +27,6 @@ public class Build {
/** Value used for when a build property is unknown. */ /** Value used for when a build property is unknown. */
public static final String UNKNOWN = "unknown"; public static final String UNKNOWN = "unknown";
/**
* Since there might be a case where new versions of the cm framework use applications running
* old versions of the protocol (and thus old versions of this class), we need a versioning
* system for the parcels sent between the core framework and its sdk users.
*
* This parcelable version should be the latest version API version listed in
* {@link CM_VERSION_CODES}
* @hide
*/
public static final int PARCELABLE_VERSION = CM_VERSION_CODES.ELDERBERRY;
/** A build ID utilized to distinguish cyanogenmod versions */ /** A build ID utilized to distinguish cyanogenmod versions */
public static final String CYANOGENMOD_VERSION = "ro.cm.version"; public static final String CYANOGENMOD_VERSION = "ro.cm.version";

View File

@ -0,0 +1,150 @@
/*
* Copyright (C) 2016 The CyanogenMod Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package cyanogenmod.os;
import android.os.Parcel;
import cyanogenmod.os.Build.CM_VERSION_CODES;
/**
* Simply, Concierge handles your parcels and makes sure they get marshalled and unmarshalled
* correctly when cross IPC boundaries even when there is a version mismatch between the client
* sdk level and the framework implementation.
*
* <p>On incoming parcel (to be unmarshalled):
*
* <pre class="prettyprint">
* ParcelInfo incomingParcelInfo = Concierge.receiveParcel(incomingParcel);
* int parcelableVersion = incomingParcelInfo.getParcelVersion();
*
* // Do unmarshalling steps here iterating over every plausible version
*
* // Complete the process
* incomingParcelInfo.complete();
* </pre>
*
* <p>On outgoing parcel (to be marshalled):
*
* <pre class="prettyprint">
* ParcelInfo outgoingParcelInfo = Concierge.prepareParcel(incomingParcel);
*
* // Do marshalling steps here iterating over every plausible version
*
* // Complete the process
* outgoingParcelInfo.complete();
* </pre>
*
* {@hide} // Unhide?
*/
public final class Concierge {
/** Not instantiable */
private Concierge() {
// Don't instantiate
}
/**
* Since there might be a case where new versions of the cm framework use applications running
* old versions of the protocol (and thus old versions of this class), we need a versioning
* system for the parcels sent between the core framework and its sdk users.
*
* This parcelable version should be the latest version API version listed in
* {@link CM_VERSION_CODES}
* @hide
*/
public static final int PARCELABLE_VERSION = CM_VERSION_CODES.ELDERBERRY;
/**
* Tell the concierge to receive our parcel, so we can get information from it.
*
* MUST CALL {@link ParcelInfo#complete()} AFTER UNMARSHALLING.
*
* @param parcel Incoming parcel to be unmarshalled
* @return {@link ParcelInfo} containing parcel information, specifically the version.
*/
public static ParcelInfo receiveParcel(Parcel parcel) {
return new ParcelInfo(parcel);
}
/**
* Prepare a parcel for the Concierge.
*
* MUST CALL {@link ParcelInfo#complete()} AFTER MARSHALLING.
*
* @param parcel Outgoing parcel to be marshalled
* @return {@link ParcelInfo} containing parcel information, specifically the version.
*/
public static ParcelInfo prepareParcel(Parcel parcel) {
return new ParcelInfo(parcel, PARCELABLE_VERSION);
}
public final static class ParcelInfo {
private Parcel mParcel;
private int mParcelableVersion;
private int mParcelableSize;
private int mStartPosition;
private int mSizePosition;
private boolean mCreation = false;
ParcelInfo(Parcel parcel) {
mCreation = false;
mParcel = parcel;
mParcelableVersion = parcel.readInt();
mParcelableSize = parcel.readInt();
mStartPosition = parcel.dataPosition();
}
ParcelInfo(Parcel parcel, int parcelableVersion) {
mCreation = true;
mParcel = parcel;
mParcelableVersion = parcelableVersion;
// Write parcelable version, make sure to define explicit changes
// within {@link #PARCELABLE_VERSION);
mParcel.writeInt(mParcelableVersion);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
mSizePosition = parcel.dataPosition();
mParcel.writeInt(0);
mStartPosition = parcel.dataPosition();
}
/**
* Get the parcel version from the {@link Parcel} received by the Concierge.
* @return {@link #PARCELABLE_VERSION} of the {@link Parcel}
*/
public int getParcelVersion() {
return mParcelableVersion;
}
/**
* Complete the {@link ParcelInfo} for the Concierge.
*/
public void complete() {
if (mCreation) {
// Go back and write size
mParcelableSize = mParcel.dataPosition() - mStartPosition;
mParcel.setDataPosition(mSizePosition);
mParcel.writeInt(mParcelableSize);
mParcel.setDataPosition(mStartPosition + mParcelableSize);
} else {
mParcel.setDataPosition(mStartPosition + mParcelableSize);
}
}
}
}

View File

@ -23,6 +23,9 @@ import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
import org.xmlpull.v1.XmlPullParser; import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException; import org.xmlpull.v1.XmlPullParserException;
@ -188,35 +191,23 @@ public final class AirplaneModeSettings implements Parcelable {
/** @hide */ /** @hide */
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
dest.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
// === BOYSENBERRY === // === BOYSENBERRY ===
dest.writeInt(mOverride ? 1 : 0); dest.writeInt(mOverride ? 1 : 0);
dest.writeInt(mValue); dest.writeInt(mValue);
dest.writeInt(mDirty ? 1 : 0); dest.writeInt(mDirty ? 1 : 0);
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = dest.dataPosition() - startPosition; parcelInfo.complete();
dest.setDataPosition(sizePosition);
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
/** @hide */ /** @hide */
public void readFromParcel(Parcel in) { public void readFromParcel(Parcel in) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(in);
int parcelableVersion = in.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = in.readInt();
int startPosition = in.dataPosition();
// Pattern here is that all new members should be added to the end of // Pattern here is that all new members should be added to the end of
// the writeToParcel method. Then we step through each version, until the latest // the writeToParcel method. Then we step through each version, until the latest
@ -227,6 +218,7 @@ public final class AirplaneModeSettings implements Parcelable {
mDirty = in.readInt() != 0; mDirty = in.readInt() != 0;
} }
in.setDataPosition(startPosition + parcelableSize); // Complete parcel info for the concierge
parcelInfo.complete();
} }
} }

View File

@ -22,6 +22,9 @@ import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
import org.xmlpull.v1.XmlPullParser; import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException; import org.xmlpull.v1.XmlPullParserException;
@ -186,35 +189,23 @@ public final class BrightnessSettings implements Parcelable {
/** @hide */ /** @hide */
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
dest.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
// === BOYSENBERRY === // === BOYSENBERRY ===
dest.writeInt(mOverride ? 1 : 0); dest.writeInt(mOverride ? 1 : 0);
dest.writeInt(mValue); dest.writeInt(mValue);
dest.writeInt(mDirty ? 1 : 0); dest.writeInt(mDirty ? 1 : 0);
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = dest.dataPosition() - startPosition; parcelInfo.complete();
dest.setDataPosition(sizePosition);
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
/** @hide */ /** @hide */
public void readFromParcel(Parcel in) { public void readFromParcel(Parcel in) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(in);
int parcelableVersion = in.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = in.readInt();
int startPosition = in.dataPosition();
// Pattern here is that all new members should be added to the end of // Pattern here is that all new members should be added to the end of
// the writeToParcel method. Then we step through each version, until the latest // the writeToParcel method. Then we step through each version, until the latest
@ -225,6 +216,7 @@ public final class BrightnessSettings implements Parcelable {
mDirty = in.readInt() != 0; mDirty = in.readInt() != 0;
} }
in.setDataPosition(startPosition + parcelableSize); // Complete parcel info for the concierge
parcelInfo.complete();
} }
} }

View File

@ -32,6 +32,9 @@ import android.telephony.TelephonyManager;
import com.android.internal.telephony.RILConstants; import com.android.internal.telephony.RILConstants;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
import org.xmlpull.v1.XmlPullParser; import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException; import org.xmlpull.v1.XmlPullParserException;
@ -426,15 +429,8 @@ public final class ConnectionSettings implements Parcelable {
/** @hide */ /** @hide */
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
dest.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
// === BOYSENBERRY === // === BOYSENBERRY ===
dest.writeInt(mConnectionId); dest.writeInt(mConnectionId);
@ -447,20 +443,15 @@ public final class ConnectionSettings implements Parcelable {
dest.writeInt(mSubId); dest.writeInt(mSubId);
} }
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = dest.dataPosition() - startPosition; parcelInfo.complete();
dest.setDataPosition(sizePosition);
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
/** @hide */ /** @hide */
public void readFromParcel(Parcel in) { public void readFromParcel(Parcel in) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(in);
int parcelableVersion = in.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = in.readInt();
int startPosition = in.dataPosition();
// Pattern here is that all new members should be added to the end of // Pattern here is that all new members should be added to the end of
// the writeToParcel method. Then we step through each version, until the latest // the writeToParcel method. Then we step through each version, until the latest
@ -478,6 +469,7 @@ public final class ConnectionSettings implements Parcelable {
} }
} }
in.setDataPosition(startPosition + parcelableSize); // Complete parcel info for the concierge
parcelInfo.complete();
} }
} }

View File

@ -21,14 +21,15 @@ import android.content.Context;
import android.os.Parcel; import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import android.os.RemoteException; import android.os.RemoteException;
import android.os.ServiceManager;
import android.util.Log; import android.util.Log;
import android.view.WindowManagerPolicy;
/* import android.view.WindowManagerPolicyControl; */ /* import android.view.WindowManagerPolicyControl; */
import com.android.internal.policy.IKeyguardService; import com.android.internal.policy.IKeyguardService;
/* import com.android.internal.policy.PolicyManager; */ /* import com.android.internal.policy.PolicyManager; */
import cyanogenmod.app.Profile; import cyanogenmod.app.Profile;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
/** /**
* The {@link LockSettings} class allows for overriding and setting the * The {@link LockSettings} class allows for overriding and setting the
@ -147,34 +148,22 @@ public final class LockSettings implements Parcelable {
/** @hide */ /** @hide */
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
dest.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
// === BOYSENBERRY === // === BOYSENBERRY ===
dest.writeInt(mValue); dest.writeInt(mValue);
dest.writeInt(mDirty ? 1 : 0); dest.writeInt(mDirty ? 1 : 0);
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = dest.dataPosition() - startPosition; parcelInfo.complete();
dest.setDataPosition(sizePosition);
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
/** @hide */ /** @hide */
public void readFromParcel(Parcel in) { public void readFromParcel(Parcel in) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(in);
int parcelableVersion = in.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = in.readInt();
int startPosition = in.dataPosition();
// Pattern here is that all new members should be added to the end of // Pattern here is that all new members should be added to the end of
// the writeToParcel method. Then we step through each version, until the latest // the writeToParcel method. Then we step through each version, until the latest
@ -184,6 +173,7 @@ public final class LockSettings implements Parcelable {
mDirty = in.readInt() != 0; mDirty = in.readInt() != 0;
} }
in.setDataPosition(startPosition + parcelableSize); // Complete parcel info for the concierge
parcelInfo.complete();
} }
} }

View File

@ -22,6 +22,9 @@ import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
import org.xmlpull.v1.XmlPullParser; import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException; import org.xmlpull.v1.XmlPullParserException;
@ -178,35 +181,23 @@ public final class RingModeSettings implements Parcelable {
/** @hide */ /** @hide */
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
dest.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
// === BOYSENBERRY === // === BOYSENBERRY ===
dest.writeInt(mOverride ? 1 : 0); dest.writeInt(mOverride ? 1 : 0);
dest.writeString(mValue); dest.writeString(mValue);
dest.writeInt(mDirty ? 1 : 0); dest.writeInt(mDirty ? 1 : 0);
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = dest.dataPosition() - startPosition; parcelInfo.complete();
dest.setDataPosition(sizePosition);
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
/** @hide */ /** @hide */
public void readFromParcel(Parcel in) { public void readFromParcel(Parcel in) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(in);
int parcelableVersion = in.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = in.readInt();
int startPosition = in.dataPosition();
// Pattern here is that all new members should be added to the end of // Pattern here is that all new members should be added to the end of
// the writeToParcel method. Then we step through each version, until the latest // the writeToParcel method. Then we step through each version, until the latest
@ -217,6 +208,7 @@ public final class RingModeSettings implements Parcelable {
mDirty = in.readInt() != 0; mDirty = in.readInt() != 0;
} }
in.setDataPosition(startPosition + parcelableSize); // Complete parcel info for the concierge
parcelInfo.complete();
} }
} }

View File

@ -16,16 +16,20 @@
package cyanogenmod.profiles; package cyanogenmod.profiles;
import cyanogenmod.os.Build;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException;
import android.content.Context; import android.content.Context;
import android.os.Parcel; import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
import java.io.IOException; import java.io.IOException;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException;
/** /**
* The {@link StreamSettings} class allows for creating various {@link android.media.AudioManager} * The {@link StreamSettings} class allows for creating various {@link android.media.AudioManager}
* overrides on the device depending on their capabilities. * overrides on the device depending on their capabilities.
@ -177,15 +181,8 @@ public final class StreamSettings implements Parcelable{
/** @hide */ /** @hide */
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
dest.writeInt(Build.PARCELABLE_VERSION);
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
// === BOYSENBERRY === // === BOYSENBERRY ===
dest.writeInt(mStreamId); dest.writeInt(mStreamId);
@ -193,20 +190,15 @@ public final class StreamSettings implements Parcelable{
dest.writeInt(mValue); dest.writeInt(mValue);
dest.writeInt(mDirty ? 1 : 0); dest.writeInt(mDirty ? 1 : 0);
// Go back and write size // Complete the parcel info for the concierge
int parcelableSize = dest.dataPosition() - startPosition; parcelInfo.complete();
dest.setDataPosition(sizePosition);
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
/** @hide */ /** @hide */
public void readFromParcel(Parcel in) { public void readFromParcel(Parcel in) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(in);
int parcelableVersion = in.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int parcelableSize = in.readInt();
int startPosition = in.dataPosition();
// Pattern here is that all new members should be added to the end of // Pattern here is that all new members should be added to the end of
// the writeToParcel method. Then we step through each version, until the latest // the writeToParcel method. Then we step through each version, until the latest
@ -218,6 +210,7 @@ public final class StreamSettings implements Parcelable{
mDirty = in.readInt() != 0; mDirty = in.readInt() != 0;
} }
in.setDataPosition(startPosition + parcelableSize); // Complete parcel info for the concierge
parcelInfo.complete();
} }
} }

View File

@ -19,7 +19,9 @@ import android.content.pm.ThemeUtils;
import android.content.res.ThemeConfig; import android.content.res.ThemeConfig;
import android.os.Parcel; import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
import java.util.Collections; import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
@ -124,11 +126,9 @@ public final class ThemeChangeRequest implements Parcelable {
} }
private ThemeChangeRequest(Parcel source) { private ThemeChangeRequest(Parcel source) {
// Read parcelable version, make sure to define explicit changes // Read parcelable version via the Concierge
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.receiveParcel(source);
int version = source.readInt(); int parcelableVersion = parcelInfo.getParcelVersion();
int size = source.readInt();
int start = source.dataPosition();
int numComponents = source.readInt(); int numComponents = source.readInt();
for (int i = 0; i < numComponents; i++) { for (int i = 0; i < numComponents; i++) {
@ -141,7 +141,9 @@ public final class ThemeChangeRequest implements Parcelable {
} }
mRequestType = RequestType.values()[source.readInt()]; mRequestType = RequestType.values()[source.readInt()];
mWallpaperId = source.readLong(); mWallpaperId = source.readLong();
source.setDataPosition(start + size);
// Complete parcel info for the concierge
parcelInfo.complete();
} }
@Override @Override
@ -151,14 +153,8 @@ public final class ThemeChangeRequest implements Parcelable {
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
// Write parcelable version, make sure to define explicit changes // Tell the concierge to prepare the parcel
// within {@link Build.PARCELABLE_VERSION); ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
dest.writeInt(Build.PARCELABLE_VERSION);
int sizePos = dest.dataPosition();
// Inject a placeholder that will store the parcel size from this point on
// (not including the size itself).
dest.writeInt(0);
int dataStartPos = dest.dataPosition();
dest.writeInt(mThemeComponents.size()); dest.writeInt(mThemeComponents.size());
for (String component : mThemeComponents.keySet()) { for (String component : mThemeComponents.keySet()) {
@ -173,11 +169,8 @@ public final class ThemeChangeRequest implements Parcelable {
dest.writeInt(mRequestType.ordinal()); dest.writeInt(mRequestType.ordinal());
dest.writeLong(mWallpaperId); dest.writeLong(mWallpaperId);
// Go back and write size // Complete the parcel info for the concierge
int size = dest.dataPosition() - dataStartPos; parcelInfo.complete();
dest.setDataPosition(sizePos);
dest.writeInt(size);
dest.setDataPosition(dataStartPos + size);
} }
public static final Parcelable.Creator<ThemeChangeRequest> CREATOR = public static final Parcelable.Creator<ThemeChangeRequest> CREATOR =

View File

@ -19,7 +19,10 @@ package cyanogenmod.weather;
import android.location.Location; import android.location.Location;
import android.os.Parcel; import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
import cyanogenmod.providers.WeatherContract; import cyanogenmod.providers.WeatherContract;
/** /**
@ -164,9 +167,10 @@ public final class RequestInfo implements Parcelable {
} }
private RequestInfo(Parcel parcel) { private RequestInfo(Parcel parcel) {
int parcelableVersion = parcel.readInt(); // Read parcelable version via the Concierge
int parcelableSize = parcel.readInt(); ParcelInfo parcelInfo = Concierge.receiveParcel(parcel);
int startPosition = parcel.dataPosition(); int parcelableVersion = parcelInfo.getParcelVersion();
if (parcelableVersion >= Build.CM_VERSION_CODES.ELDERBERRY) { if (parcelableVersion >= Build.CM_VERSION_CODES.ELDERBERRY) {
mKey = parcel.readInt(); mKey = parcel.readInt();
mRequestType = parcel.readInt(); mRequestType = parcel.readInt();
@ -186,7 +190,9 @@ public final class RequestInfo implements Parcelable {
mIsQueryOnly = (parcel.readInt() == 1); mIsQueryOnly = (parcel.readInt() == 1);
mListener = IRequestInfoListener.Stub.asInterface(parcel.readStrongBinder()); mListener = IRequestInfoListener.Stub.asInterface(parcel.readStrongBinder());
} }
parcel.setDataPosition(startPosition + parcelableSize);
// Complete parcel info for the concierge
parcelInfo.complete();
} }
@ -274,11 +280,8 @@ public final class RequestInfo implements Parcelable {
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
dest.writeInt(Build.PARCELABLE_VERSION); // Tell the concierge to prepare the parcel
ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
// ==== ELDERBERRY ===== // ==== ELDERBERRY =====
dest.writeInt(mKey); dest.writeInt(mKey);
@ -299,10 +302,8 @@ public final class RequestInfo implements Parcelable {
dest.writeInt(mIsQueryOnly == true ? 1 : 0); dest.writeInt(mIsQueryOnly == true ? 1 : 0);
dest.writeStrongBinder(mListener.asBinder()); dest.writeStrongBinder(mListener.asBinder());
int parcelableSize = dest.dataPosition() - startPosition; // Complete parcel info for the concierge
dest.setDataPosition(sizePosition); parcelInfo.complete();
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
@Override @Override

View File

@ -19,7 +19,10 @@ package cyanogenmod.weather;
import android.annotation.NonNull; import android.annotation.NonNull;
import android.os.Parcel; import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
import cyanogenmod.providers.WeatherContract; import cyanogenmod.providers.WeatherContract;
import cyanogenmod.weatherservice.ServiceRequest; import cyanogenmod.weatherservice.ServiceRequest;
import cyanogenmod.weatherservice.ServiceRequestResult; import cyanogenmod.weatherservice.ServiceRequestResult;
@ -261,9 +264,10 @@ public final class WeatherInfo implements Parcelable {
} }
private WeatherInfo(Parcel parcel) { private WeatherInfo(Parcel parcel) {
int parcelableVersion = parcel.readInt(); // Read parcelable version via the Concierge
int parcelableSize = parcel.readInt(); ParcelInfo parcelInfo = Concierge.receiveParcel(parcel);
int startPosition = parcel.dataPosition(); int parcelableVersion = parcelInfo.getParcelVersion();
if (parcelableVersion >= Build.CM_VERSION_CODES.ELDERBERRY) { if (parcelableVersion >= Build.CM_VERSION_CODES.ELDERBERRY) {
mKey = parcel.readInt(); mKey = parcel.readInt();
mCityId = parcel.readString(); mCityId = parcel.readString();
@ -283,7 +287,9 @@ public final class WeatherInfo implements Parcelable {
forecastListSize--; forecastListSize--;
} }
} }
parcel.setDataPosition(startPosition + parcelableSize);
// Complete parcel info for the concierge
parcelInfo.complete();
} }
@Override @Override
@ -293,11 +299,8 @@ public final class WeatherInfo implements Parcelable {
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
dest.writeInt(Build.PARCELABLE_VERSION); // Tell the concierge to prepare the parcel
ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
// ==== ELDERBERRY ===== // ==== ELDERBERRY =====
dest.writeInt(mKey); dest.writeInt(mKey);
@ -316,10 +319,8 @@ public final class WeatherInfo implements Parcelable {
dayForecast.writeToParcel(dest, 0); dayForecast.writeToParcel(dest, 0);
} }
int parcelableSize = dest.dataPosition() - startPosition; // Complete parcel info for the concierge
dest.setDataPosition(sizePosition); parcelInfo.complete();
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
public static final Parcelable.Creator<WeatherInfo> CREATOR = public static final Parcelable.Creator<WeatherInfo> CREATOR =
@ -414,11 +415,8 @@ public final class WeatherInfo implements Parcelable {
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
dest.writeInt(Build.PARCELABLE_VERSION); // Tell the concierge to prepare the parcel
ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
// ==== ELDERBERRY ===== // ==== ELDERBERRY =====
dest.writeInt(mKey); dest.writeInt(mKey);
@ -426,10 +424,8 @@ public final class WeatherInfo implements Parcelable {
dest.writeFloat(mHigh); dest.writeFloat(mHigh);
dest.writeInt(mConditionCode); dest.writeInt(mConditionCode);
int parcelableSize = dest.dataPosition() - startPosition; // Complete parcel info for the concierge
dest.setDataPosition(sizePosition); parcelInfo.complete();
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
public static final Parcelable.Creator<DayForecast> CREATOR = public static final Parcelable.Creator<DayForecast> CREATOR =
@ -446,16 +442,19 @@ public final class WeatherInfo implements Parcelable {
}; };
private DayForecast(Parcel parcel) { private DayForecast(Parcel parcel) {
int parcelableVersion = parcel.readInt(); // Read parcelable version via the Concierge
int parcelableSize = parcel.readInt(); ParcelInfo parcelInfo = Concierge.receiveParcel(parcel);
int startPosition = parcel.dataPosition(); int parcelableVersion = parcelInfo.getParcelVersion();
if (parcelableVersion >= Build.CM_VERSION_CODES.ELDERBERRY) { if (parcelableVersion >= Build.CM_VERSION_CODES.ELDERBERRY) {
mKey = parcel.readInt(); mKey = parcel.readInt();
mLow = parcel.readFloat(); mLow = parcel.readFloat();
mHigh = parcel.readFloat(); mHigh = parcel.readFloat();
mConditionCode = parcel.readInt(); mConditionCode = parcel.readInt();
} }
parcel.setDataPosition(startPosition + parcelableSize);
// Complete parcel info for the concierge
parcelInfo.complete();
} }
@Override @Override

View File

@ -18,7 +18,10 @@ package cyanogenmod.weather;
import android.os.Parcel; import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
/** /**
* A class representing a geographical location that a weather service provider can use to * A class representing a geographical location that a weather service provider can use to
@ -92,9 +95,10 @@ public final class WeatherLocation implements Parcelable{
} }
private WeatherLocation(Parcel in) { private WeatherLocation(Parcel in) {
int parcelableVersion = in.readInt(); // Read parcelable version via the Concierge
int parcelableSize = in.readInt(); ParcelInfo parcelInfo = Concierge.receiveParcel(in);
int startPosition = in.dataPosition(); int parcelableVersion = parcelInfo.getParcelVersion();
if (parcelableVersion >= Build.CM_VERSION_CODES.ELDERBERRY) { if (parcelableVersion >= Build.CM_VERSION_CODES.ELDERBERRY) {
mKey = in.readInt(); mKey = in.readInt();
mCityId = in.readString(); mCityId = in.readString();
@ -103,7 +107,9 @@ public final class WeatherLocation implements Parcelable{
mCountryId = in.readString(); mCountryId = in.readString();
mCountry = in.readString(); mCountry = in.readString();
} }
in.setDataPosition(startPosition + parcelableSize);
// Complete parcel info for the concierge
parcelInfo.complete();
} }
public static final Creator<WeatherLocation> CREATOR = new Creator<WeatherLocation>() { public static final Creator<WeatherLocation> CREATOR = new Creator<WeatherLocation>() {
@ -125,11 +131,8 @@ public final class WeatherLocation implements Parcelable{
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
dest.writeInt(Build.PARCELABLE_VERSION); // Tell the concierge to prepare the parcel
ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
// ==== ELDERBERRY ===== // ==== ELDERBERRY =====
dest.writeInt(mKey); dest.writeInt(mKey);
@ -139,10 +142,8 @@ public final class WeatherLocation implements Parcelable{
dest.writeString(mCountryId); dest.writeString(mCountryId);
dest.writeString(mCountry); dest.writeString(mCountry);
int parcelableSize = dest.dataPosition() - startPosition; // Complete parcel info for the concierge
dest.setDataPosition(sizePosition); parcelInfo.complete();
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
@Override @Override

View File

@ -19,7 +19,10 @@ package cyanogenmod.weatherservice;
import android.annotation.NonNull; import android.annotation.NonNull;
import android.os.Parcel; import android.os.Parcel;
import android.os.Parcelable; import android.os.Parcelable;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import cyanogenmod.os.Concierge;
import cyanogenmod.os.Concierge.ParcelInfo;
import cyanogenmod.weather.WeatherLocation; import cyanogenmod.weather.WeatherLocation;
import cyanogenmod.weather.WeatherInfo; import cyanogenmod.weather.WeatherInfo;
@ -37,9 +40,10 @@ public final class ServiceRequestResult implements Parcelable {
private ServiceRequestResult() {} private ServiceRequestResult() {}
private ServiceRequestResult(Parcel in) { private ServiceRequestResult(Parcel in) {
int parcelableVersion = in.readInt(); // Read parcelable version via the Concierge
int parcelableSize = in.readInt(); ParcelInfo parcelInfo = Concierge.receiveParcel(in);
int startPosition = in.dataPosition(); int parcelableVersion = parcelInfo.getParcelVersion();
if (parcelableVersion >= Build.CM_VERSION_CODES.ELDERBERRY) { if (parcelableVersion >= Build.CM_VERSION_CODES.ELDERBERRY) {
mKey = in.readInt(); mKey = in.readInt();
int hasWeatherInfo = in.readInt(); int hasWeatherInfo = in.readInt();
@ -56,7 +60,9 @@ public final class ServiceRequestResult implements Parcelable {
} }
} }
} }
in.setDataPosition(startPosition + parcelableSize);
// Complete parcel info for the concierge
parcelInfo.complete();
} }
public static final Creator<ServiceRequestResult> CREATOR public static final Creator<ServiceRequestResult> CREATOR
@ -79,11 +85,8 @@ public final class ServiceRequestResult implements Parcelable {
@Override @Override
public void writeToParcel(Parcel dest, int flags) { public void writeToParcel(Parcel dest, int flags) {
dest.writeInt(Build.PARCELABLE_VERSION); // Tell the concierge to prepare the parcel
ParcelInfo parcelInfo = Concierge.prepareParcel(dest);
int sizePosition = dest.dataPosition();
dest.writeInt(0);
int startPosition = dest.dataPosition();
// ==== ELDERBERRY ===== // ==== ELDERBERRY =====
dest.writeInt(mKey); dest.writeInt(mKey);
@ -103,10 +106,8 @@ public final class ServiceRequestResult implements Parcelable {
dest.writeInt(0); dest.writeInt(0);
} }
int parcelableSize = dest.dataPosition() - startPosition; // Complete parcel info for the concierge
dest.setDataPosition(sizePosition); parcelInfo.complete();
dest.writeInt(parcelableSize);
dest.setDataPosition(startPosition + parcelableSize);
} }
public static class Builder { public static class Builder {

View File

@ -20,6 +20,7 @@ import android.test.suitebuilder.annotation.SmallTest;
import cyanogenmod.os.Build; import cyanogenmod.os.Build;
import android.test.AndroidTestCase; import android.test.AndroidTestCase;
import cyanogenmod.os.Concierge;
/** /**
* Created by adnan on 7/14/15. * Created by adnan on 7/14/15.
@ -38,20 +39,20 @@ public class BuildTest extends AndroidTestCase {
@SmallTest @SmallTest
public void testSdkApiLevelCurrent() { public void testSdkApiLevelCurrent() {
assertEquals(Build.PARCELABLE_VERSION, Build.CM_VERSION.SDK_INT); assertEquals(Concierge.PARCELABLE_VERSION, Build.CM_VERSION.SDK_INT);
} }
@SmallTest @SmallTest
public void testSdkApiLevelCanMatch() { public void testSdkApiLevelCanMatch() {
String apiName = Build.getNameForSDKInt(Build.CM_VERSION.SDK_INT); String apiName = Build.getNameForSDKInt(Build.CM_VERSION.SDK_INT);
assertNotNull(apiName); assertNotNull(apiName);
assertEquals(Build.getNameForSDKInt(Build.PARCELABLE_VERSION), apiName); assertEquals(Build.getNameForSDKInt(Concierge.PARCELABLE_VERSION), apiName);
} }
@SmallTest @SmallTest
public void testSdkApiLevelSkippedIfGreaterThanAllowed() { public void testSdkApiLevelSkippedIfGreaterThanAllowed() {
int i = 0; int i = 0;
if (Build.CM_VERSION.SDK_INT > Build.PARCELABLE_VERSION + 1) { if (Build.CM_VERSION.SDK_INT > Concierge.PARCELABLE_VERSION + 1) {
i++; i++;
} }
assertEquals(0, i); assertEquals(0, i);