/* * Copyright (C) 2013 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 LOG_TAG "CameraTraces" #define ATRACE_TAG ATRACE_TAG_CAMERA //#define LOG_NDEBUG 0 #include "utils/CameraTraces.h" #include <utils/ProcessCallStack.h> #include <utils/Mutex.h> #include <utils/List.h> #include <utils/Log.h> #include <cutils/trace.h> namespace android { namespace camera3 { struct CameraTracesImpl { Mutex tracesLock; List<ProcessCallStack> pcsList; }; // class CameraTraces::Impl; static CameraTracesImpl gImpl; CameraTracesImpl& CameraTraces::sImpl = gImpl; void CameraTraces::saveTrace() { ALOGV("%s: begin", __FUNCTION__); ATRACE_BEGIN("CameraTraces::saveTrace"); Mutex::Autolock al(sImpl.tracesLock); List<ProcessCallStack>& pcsList = sImpl.pcsList; // Insert new ProcessCallStack, and immediately crawl all the threads pcsList.push_front(ProcessCallStack()); ProcessCallStack& pcs = *pcsList.begin(); pcs.update(); if (pcsList.size() > MAX_TRACES) { // Prune list periodically and discard oldest entry pcsList.erase(--pcsList.end()); } IF_ALOGV() { pcs.log(LOG_TAG, ANDROID_LOG_VERBOSE); } ALOGD("Process trace saved. Use dumpsys media.camera to view."); ATRACE_END(); } status_t CameraTraces::dump(int fd, const Vector<String16> &args __attribute__((unused))) { ALOGV("%s: fd = %d", __FUNCTION__, fd); Mutex::Autolock al(sImpl.tracesLock); List<ProcessCallStack>& pcsList = sImpl.pcsList; if (fd < 0) { ALOGW("%s: Negative FD (%d)", __FUNCTION__, fd); return BAD_VALUE; } dprintf(fd, "== Camera error traces (%zu): ==\n", pcsList.size()); if (pcsList.empty()) { dprintf(fd, " No camera traces collected.\n"); } // Print newest items first List<ProcessCallStack>::iterator it, end; for (it = pcsList.begin(), end = pcsList.end(); it != end; ++it) { const ProcessCallStack& pcs = *it; pcs.dump(fd, DUMP_INDENT); } return OK; } }; // namespace camera3 }; // namespace android