/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=8 sts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include #include "mozilla/IOInterposer.h" #include "mozilla/PoisonIOInterposer.h" #include "mozilla/ProcessedStack.h" #include "mozilla/SHA1.h" #include "mozilla/Scoped.h" #include "mozilla/StaticPtr.h" #include "mozilla/Telemetry.h" #include "mozilla/Unused.h" #include "nsAppDirectoryServiceDefs.h" #include "nsDirectoryServiceUtils.h" #include "nsLocalFile.h" #include "nsPrintfCString.h" #include "mozilla/StackWalk.h" #include "plstr.h" #include "prio.h" #ifdef XP_WIN # define NS_SLASH "\\" # include # include # include # include # include # include #else # define NS_SLASH "/" #endif #include "LateWriteChecks.h" /*************************** Auxiliary Declarations ***************************/ static MOZ_THREAD_LOCAL(int) tlsSuspendLateWriteChecks; bool SuspendingLateWriteChecksForCurrentThread() { if (!tlsSuspendLateWriteChecks.init()) { return true; } return tlsSuspendLateWriteChecks.get() > 0; } // This a wrapper over a file descriptor that provides a Printf method and // computes the sha1 of the data that passes through it. class SHA1Stream { public: explicit SHA1Stream(FILE* aStream) : mFile(aStream) { MozillaRegisterDebugFILE(mFile); } void Printf(const char* aFormat, ...) MOZ_FORMAT_PRINTF(2, 3) { MOZ_ASSERT(mFile); va_list list; va_start(list, aFormat); nsAutoCString str; str.AppendVprintf(aFormat, list); va_end(list); mSHA1.update(str.get(), str.Length()); mozilla::Unused << fwrite(str.get(), 1, str.Length(), mFile); } void Finish(mozilla::SHA1Sum::Hash& aHash) { int fd = fileno(mFile); fflush(mFile); MozillaUnRegisterDebugFD(fd); fclose(mFile); mSHA1.finish(aHash); mFile = nullptr; } private: FILE* mFile; mozilla::SHA1Sum mSHA1; }; static void RecordStackWalker(uint32_t aFrameNumber, void* aPC, void* aSP, void* aClosure) { std::vector* stack = static_cast*>(aClosure); stack->push_back(reinterpret_cast(aPC)); } /**************************** Late-Write Observer ****************************/ /** * An implementation of IOInterposeObserver to be registered with IOInterposer. * This observer logs all writes as late writes. */ class LateWriteObserver final : public mozilla::IOInterposeObserver { using char_type = mozilla::filesystem::Path::value_type; public: explicit LateWriteObserver(const char_type* aProfileDirectory) : mProfileDirectory(NS_xstrdup(aProfileDirectory)) {} ~LateWriteObserver() { free(mProfileDirectory); mProfileDirectory = nullptr; } void Observe( mozilla::IOInterposeObserver::Observation& aObservation) override; private: char_type* mProfileDirectory; }; void LateWriteObserver::Observe( mozilla::IOInterposeObserver::Observation& aOb) { if (SuspendingLateWriteChecksForCurrentThread()) { return; } #ifdef DEBUG MOZ_CRASH(); #endif // If we can't record then abort if (!mozilla::Telemetry::CanRecordExtended()) { return; } // Write the stack and loaded libraries to a file. We can get here // concurrently from many writes, so we use multiple temporary files. std::vector rawStack; MozStackWalk(RecordStackWalker, /* skipFrames */ 0, /* maxFrames */ 0, &rawStack); mozilla::Telemetry::ProcessedStack stack = mozilla::Telemetry::GetStackAndModules(rawStack); nsTAutoString nameAux(mProfileDirectory); nameAux.AppendLiteral(NS_SLASH "Telemetry.LateWriteTmpXXXXXX"); char_type* name = nameAux.BeginWriting(); // We want the sha1 of the entire file, so please don't write to fd // directly; use sha1Stream. FILE* stream; #ifdef XP_WIN HANDLE hFile; do { // mkstemp isn't supported so keep trying until we get a file _wmktemp_s(char16ptr_t(name), NS_strlen(name) + 1); hFile = CreateFileW(char16ptr_t(name), GENERIC_WRITE, 0, nullptr, CREATE_NEW, FILE_ATTRIBUTE_NORMAL, nullptr); } while (GetLastError() == ERROR_FILE_EXISTS); if (hFile == INVALID_HANDLE_VALUE) { MOZ_CRASH("Um, how did we get here?"); } // http://support.microsoft.com/kb/139640 int fd = _open_osfhandle((intptr_t)hFile, _O_APPEND); if (fd == -1) { MOZ_CRASH("Um, how did we get here?"); } stream = _fdopen(fd, "w"); #else int fd = mkstemp(name); if (fd == -1) { MOZ_CRASH("mkstemp failed"); } stream = fdopen(fd, "w"); #endif SHA1Stream sha1Stream(stream); size_t numModules = stack.GetNumModules(); sha1Stream.Printf("%u\n", (unsigned)numModules); for (size_t i = 0; i < numModules; ++i) { mozilla::Telemetry::ProcessedStack::Module module = stack.GetModule(i); sha1Stream.Printf("%s %s\n", module.mBreakpadId.get(), NS_ConvertUTF16toUTF8(module.mName).get()); } size_t numFrames = stack.GetStackSize(); sha1Stream.Printf("%u\n", (unsigned)numFrames); for (size_t i = 0; i < numFrames; ++i) { const mozilla::Telemetry::ProcessedStack::Frame& frame = stack.GetFrame(i); // NOTE: We write the offsets, while the atos tool expects a value with // the virtual address added. For example, running otool -l on the the // firefox binary shows // cmd LC_SEGMENT_64 // cmdsize 632 // segname __TEXT // vmaddr 0x0000000100000000 // so to print the line matching the offset 123 one has to run // atos -o firefox 0x100000123. sha1Stream.Printf("%d %x\n", frame.mModIndex, (unsigned)frame.mOffset); } mozilla::SHA1Sum::Hash sha1; sha1Stream.Finish(sha1); // Note: These files should be deleted by telemetry once it reads them. If // there were no telemetry runs by the time we shut down, we just add files // to the existing ones instead of replacing them. Given that each of these // files is a bug to be fixed, that is probably the right thing to do. // We append the sha1 of the contents to the file name. This provides a simple // client side deduplication. nsAutoString finalName(u"Telemetry.LateWriteFinal-"_ns); for (int i = 0; i < 20; ++i) { finalName.AppendPrintf("%02x", sha1[i]); } RefPtr file = new nsLocalFile(nameAux); file->RenameTo(nullptr, finalName); } /******************************* Setup/Teardown *******************************/ static mozilla::StaticAutoPtr sLateWriteObserver; namespace mozilla { void InitLateWriteChecks() { nsCOMPtr mozFile; NS_GetSpecialDirectory(NS_APP_USER_PROFILE_50_DIR, getter_AddRefs(mozFile)); if (mozFile) { PathString nativePath = mozFile->NativePath(); if (nativePath.get()) { sLateWriteObserver = new LateWriteObserver(nativePath.get()); } } } void BeginLateWriteChecks() { if (sLateWriteObserver) { IOInterposer::Register(IOInterposeObserver::OpWriteFSync, sLateWriteObserver); } } void StopLateWriteChecks() { if (sLateWriteObserver) { IOInterposer::Unregister(IOInterposeObserver::OpAll, sLateWriteObserver); // Deallocation would not be thread-safe, and StopLateWriteChecks() is // called at shutdown and only in special cases. // sLateWriteObserver = nullptr; } } void PushSuspendLateWriteChecks() { if (!tlsSuspendLateWriteChecks.init()) { return; } tlsSuspendLateWriteChecks.set(tlsSuspendLateWriteChecks.get() + 1); } void PopSuspendLateWriteChecks() { if (!tlsSuspendLateWriteChecks.init()) { return; } int current = tlsSuspendLateWriteChecks.get(); MOZ_ASSERT(current > 0); tlsSuspendLateWriteChecks.set(current - 1); } } // namespace mozilla