/* * Copyright (C) 2018 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. */ #define DEBUG false // STOPSHIP if true #include "Log.h" #include "puller_util.h" namespace android { namespace os { namespace statsd { using namespace std; /** * Process all data and merge isolated with host if necessary. * For example: * NetworkBytesAtom { * int uid = 1; * State process_state = 2; * int byte_send = 3; * int byte_recv = 4; * } * additive fields are {3, 4} * If we pulled the following events (uid1_child is an isolated uid which maps to uid1): * [uid1, fg, 100, 200] * [uid1_child, fg, 100, 200] * [uid1, bg, 100, 200] * * We want to merge them and results should be: * [uid1, fg, 200, 400] * [uid1, bg, 100, 200] * * All atoms should be of the same tagId. All fields should be present. */ void mapAndMergeIsolatedUidsToHostUid(vector>& data, const sp& uidMap, int tagId, const vector& additiveFieldsVec) { // Check the first LogEvent for attribution chain or a uid field as either all atoms with this // tagId have them or none of them do. std::pair attrIndexRange; const bool hasAttributionChain = data[0]->hasAttributionChain(&attrIndexRange); bool hasUidField = (data[0]->getUidFieldIndex() != -1); if (!hasAttributionChain && !hasUidField) { VLOG("No uid or attribution chain to merge, atom %d", tagId); return; } // 1. Map all isolated uid in-place to host uid for (shared_ptr& event : data) { if (event->GetTagId() != tagId) { ALOGE("Wrong atom. Expecting %d, got %d", tagId, event->GetTagId()); return; } if (hasAttributionChain) { vector* const fieldValues = event->getMutableValues(); for (int i = attrIndexRange.first; i <= attrIndexRange.second; i++) { FieldValue& fieldValue = fieldValues->at(i); if (isAttributionUidField(fieldValue)) { const int hostUid = uidMap->getHostUidOrSelf(fieldValue.mValue.int_value); fieldValue.mValue.setInt(hostUid); } } } else { int uidFieldIndex = event->getUidFieldIndex(); if (uidFieldIndex != -1) { Value& value = (*event->getMutableValues())[uidFieldIndex].mValue; const int hostUid = uidMap->getHostUidOrSelf(value.int_value); value.setInt(hostUid); } else { ALOGE("Malformed log, uid not found. %s", event->ToString().c_str()); } } } // 2. sort the data, bit-wise sort(data.begin(), data.end(), [](const shared_ptr& lhs, const shared_ptr& rhs) { if (lhs->size() != rhs->size()) { return lhs->size() < rhs->size(); } const std::vector& lhsValues = lhs->getValues(); const std::vector& rhsValues = rhs->getValues(); for (int i = 0; i < (int)lhs->size(); i++) { if (lhsValues[i] != rhsValues[i]) { return lhsValues[i] < rhsValues[i]; } } return false; }); vector> mergedData; const set additiveFields(additiveFieldsVec.begin(), additiveFieldsVec.end()); bool needMerge = true; // 3. do the merge. // The loop invariant is this: for every event, check if it differs on // non-additive fields, or have different attribution chain length. // If so, no need to merge, add itself to the result. // Otherwise, merge the value onto the one immediately next to it. for (int i = 0; i < (int)data.size() - 1; i++) { // Size different, must be different chains. if (data[i]->size() != data[i + 1]->size()) { mergedData.push_back(data[i]); continue; } vector* lhsValues = data[i]->getMutableValues(); vector* rhsValues = data[i + 1]->getMutableValues(); needMerge = true; for (int p = 0; p < (int)lhsValues->size(); p++) { if ((*lhsValues)[p] != (*rhsValues)[p]) { int pos = (*lhsValues)[p].mField.getPosAtDepth(0); // Differ on non-additive field, abort. if (additiveFields.find(pos) == additiveFields.end()) { needMerge = false; break; } } } if (!needMerge) { mergedData.push_back(data[i]); continue; } // This should be infrequent operation. for (int p = 0; p < (int)lhsValues->size(); p++) { int pos = (*lhsValues)[p].mField.getPosAtDepth(0); if (additiveFields.find(pos) != additiveFields.end()) { (*rhsValues)[p].mValue += (*lhsValues)[p].mValue; } } } mergedData.push_back(data.back()); data.clear(); data = mergedData; } } // namespace statsd } // namespace os } // namespace android