// Copyright (c) 2018 Google LLC. // // 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. #if defined(SPIRV_TIMER_ENABLED) #include "source/util/timer.h" #include <sys/resource.h> #include <sys/time.h> #include <iomanip> #include <iostream> #include <string> namespace spvtools { namespace utils { void PrintTimerDescription(std::ostream* out, bool measure_mem_usage) { if (out) { *out << std::setw(30) << "PASS name" << std::setw(12) << "CPU time" << std::setw(12) << "WALL time" << std::setw(12) << "USR time" << std::setw(12) << "SYS time"; if (measure_mem_usage) { *out << std::setw(12) << "RSS delta" << std::setw(16) << "PGFault delta"; } *out << std::endl; } } // Do not change the order of invoking system calls. We want to make CPU/Wall // time correct as much as possible. Calling functions to get CPU/Wall time must // closely surround the target code of measuring. void Timer::Start() { if (report_stream_) { if (getrusage(RUSAGE_SELF, &usage_before_) == -1) usage_status_ |= kGetrusageFailed; if (clock_gettime(CLOCK_MONOTONIC, &wall_before_) == -1) usage_status_ |= kClockGettimeWalltimeFailed; if (clock_gettime(CLOCK_PROCESS_CPUTIME_ID, &cpu_before_) == -1) usage_status_ |= kClockGettimeCPUtimeFailed; } } // The order of invoking system calls is important with the same reason as // Timer::Start(). void Timer::Stop() { if (report_stream_ && usage_status_ == kSucceeded) { if (clock_gettime(CLOCK_PROCESS_CPUTIME_ID, &cpu_after_) == -1) usage_status_ |= kClockGettimeCPUtimeFailed; if (clock_gettime(CLOCK_MONOTONIC, &wall_after_) == -1) usage_status_ |= kClockGettimeWalltimeFailed; if (getrusage(RUSAGE_SELF, &usage_after_) == -1) usage_status_ = kGetrusageFailed; } } void Timer::Report(const char* tag) { if (!report_stream_) return; report_stream_->precision(2); *report_stream_ << std::fixed << std::setw(30) << tag; if (usage_status_ & kClockGettimeCPUtimeFailed) *report_stream_ << std::setw(12) << "Failed"; else *report_stream_ << std::setw(12) << CPUTime(); if (usage_status_ & kClockGettimeWalltimeFailed) *report_stream_ << std::setw(12) << "Failed"; else *report_stream_ << std::setw(12) << WallTime(); if (usage_status_ & kGetrusageFailed) { *report_stream_ << std::setw(12) << "Failed" << std::setw(12) << "Failed"; if (measure_mem_usage_) { *report_stream_ << std::setw(12) << "Failed" << std::setw(12) << "Failed"; } } else { *report_stream_ << std::setw(12) << UserTime() << std::setw(12) << SystemTime(); if (measure_mem_usage_) { *report_stream_ << std::fixed << std::setw(12) << RSS() << std::setw(16) << PageFault(); } } *report_stream_ << std::endl; } } // namespace utils } // namespace spvtools #endif // defined(SPIRV_TIMER_ENABLED)