|  | /* | 
|  | ** Copyright 2016, 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 <errno.h> | 
|  | #include <pthread.h> | 
|  | #include <time.h> | 
|  |  | 
|  | #include <log/log.h> | 
|  |  | 
|  | #include "log_portability.h" | 
|  |  | 
|  | // Global default if 'last' argument in __android_log_ratelimit is NULL | 
|  | static time_t g_last_clock; | 
|  | // Global above can not deal well with callers playing games with the | 
|  | // seconds argument, so we will also hold on to the maximum value | 
|  | // ever provided and use that to gain consistency.  If the caller | 
|  | // provides their own 'last' argument, then they can play such games | 
|  | // of varying the 'seconds' argument to their pleasure. | 
|  | static time_t g_last_seconds; | 
|  | static const time_t last_seconds_default = 10; | 
|  | static const time_t last_seconds_max = 24 * 60 * 60;  // maximum of a day | 
|  | static const time_t last_seconds_min = 2;             // granularity | 
|  | // Lock to protect last_clock and last_seconds, but also 'last' | 
|  | // argument (not NULL) as supplied to __android_log_ratelimit. | 
|  | static pthread_mutex_t lock_ratelimit = PTHREAD_MUTEX_INITIALIZER; | 
|  |  | 
|  | // if last is NULL, caller _must_ provide a consistent value for | 
|  | // seconds, otherwise we will take the maximum ever issued and hold | 
|  | // on to that.  Preserves value of non-zero errno.  Return -1 if we | 
|  | // can not acquire a lock, 0 if we are not to log a message, and 1 | 
|  | // if we are ok to log a message.  Caller should check > 0 for true. | 
|  | LIBLOG_ABI_PUBLIC int __android_log_ratelimit(time_t seconds, time_t* last) { | 
|  | int save_errno = errno; | 
|  |  | 
|  | // Two reasons for trylock failure: | 
|  | //   1. In a signal handler. Must prevent deadlock | 
|  | //   2. Too many threads calling __android_log_ratelimit. | 
|  | //      Bonus to not print if they race here because that | 
|  | //      dovetails the goal of ratelimiting. One may print | 
|  | //      and the others will wait their turn ... | 
|  | if (pthread_mutex_trylock(&lock_ratelimit)) { | 
|  | if (save_errno) errno = save_errno; | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | if (seconds == 0) { | 
|  | seconds = last_seconds_default; | 
|  | } else if (seconds < last_seconds_min) { | 
|  | seconds = last_seconds_min; | 
|  | } else if (seconds > last_seconds_max) { | 
|  | seconds = last_seconds_max; | 
|  | } | 
|  |  | 
|  | if (!last) { | 
|  | if (g_last_seconds > seconds) { | 
|  | seconds = g_last_seconds; | 
|  | } else if (g_last_seconds < seconds) { | 
|  | g_last_seconds = seconds; | 
|  | } | 
|  | last = &g_last_clock; | 
|  | } | 
|  |  | 
|  | time_t now = time(NULL); | 
|  | if ((now == (time_t)-1) || ((*last + seconds) > now)) { | 
|  | pthread_mutex_unlock(&lock_ratelimit); | 
|  | if (save_errno) errno = save_errno; | 
|  | return 0; | 
|  | } | 
|  | *last = now; | 
|  | pthread_mutex_unlock(&lock_ratelimit); | 
|  | if (save_errno) errno = save_errno; | 
|  | return 1; | 
|  | } |