| /* |
| * Copyright (C) 2017 The Android Open Source 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. |
| */ |
| #include <android/os/StatsLogEventWrapper.h> |
| |
| #include <binder/Parcel.h> |
| #include <binder/Parcelable.h> |
| #include <binder/Status.h> |
| #include <utils/RefBase.h> |
| #include <vector> |
| |
| using android::Parcel; |
| using android::Parcelable; |
| using android::status_t; |
| using std::vector; |
| |
| namespace android { |
| namespace os { |
| |
| StatsLogEventWrapper::StatsLogEventWrapper(){}; |
| |
| status_t StatsLogEventWrapper::writeToParcel(Parcel* out) const { |
| // Implement me if desired. We don't currently use this. |
| ALOGE( |
| "Cannot do c++ StatsLogEventWrapper.writeToParcel(); it is not " |
| "implemented."); |
| (void)out; // To prevent compile error of unused parameter 'in' |
| return UNKNOWN_ERROR; |
| }; |
| |
| status_t StatsLogEventWrapper::readFromParcel(const Parcel* in) { |
| status_t res = OK; |
| if (in == NULL) { |
| ALOGE("statsd received parcel argument was NULL."); |
| return BAD_VALUE; |
| } |
| if ((res = in->readInt32(&mTagId)) != OK) { |
| ALOGE("statsd could not read tagId from parcel"); |
| return res; |
| } |
| if ((res = in->readInt64(&mElapsedRealTimeNs)) != OK) { |
| ALOGE("statsd could not read elapsed real time from parcel"); |
| return res; |
| } |
| if ((res = in->readInt64(&mWallClockTimeNs)) != OK) { |
| ALOGE("statsd could not read wall clock time from parcel"); |
| return res; |
| } |
| int numWorkChain = 0; |
| if ((res = in->readInt32(&numWorkChain)) != OK) { |
| ALOGE("statsd could not read number of work chains from parcel"); |
| return res; |
| } |
| if (numWorkChain > 0) { |
| for (int i = 0; i < numWorkChain; i++) { |
| int numNodes = 0; |
| if ((res = in->readInt32(&numNodes)) != OK) { |
| ALOGE( |
| "statsd could not read number of nodes in work chain from parcel"); |
| return res; |
| } |
| if (numNodes == 0) { |
| ALOGE("empty work chain"); |
| return BAD_VALUE; |
| } |
| WorkChain wc; |
| for (int j = 0; j < numNodes; j++) { |
| wc.uids.push_back(in->readInt32()); |
| wc.tags.push_back(std::string(String8(in->readString16()).string())); |
| } |
| mWorkChains.push_back(wc); |
| } |
| } |
| int dataSize = 0; |
| if ((res = in->readInt32(&dataSize)) != OK) { |
| ALOGE("statsd could not read data size from parcel"); |
| return res; |
| } |
| if (mTagId <= 0 || mElapsedRealTimeNs <= 0 || mWallClockTimeNs <= 0 || |
| dataSize <= 0) { |
| ALOGE("statsd received invalid parcel"); |
| return BAD_VALUE; |
| } |
| |
| for (int i = 0; i < dataSize; i++) { |
| int type = in->readInt32(); |
| switch (type) { |
| case StatsLogValue::INT: |
| mElements.push_back(StatsLogValue(in->readInt32())); |
| break; |
| case StatsLogValue::LONG: |
| mElements.push_back(StatsLogValue(in->readInt64())); |
| break; |
| case StatsLogValue::STRING: |
| mElements.push_back( |
| StatsLogValue(std::string(String8(in->readString16()).string()))); |
| break; |
| case StatsLogValue::FLOAT: |
| mElements.push_back(StatsLogValue(in->readFloat())); |
| break; |
| case StatsLogValue::STORAGE: |
| mElements.push_back(StatsLogValue()); |
| mElements.back().setType(StatsLogValue::STORAGE); |
| in->readByteVector(&(mElements.back().storage_value)); |
| break; |
| default: |
| ALOGE("unrecognized data type: %d", type); |
| return BAD_TYPE; |
| } |
| } |
| return NO_ERROR; |
| }; |
| |
| } // Namespace os |
| } // Namespace android |