blob: 830358699ddb8f2fa6d6fb13d3e6834b0c9b2873 [file] [log] [blame]
// Copyright 2021 The Bazel Authors. All rights reserved.
//
// 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 <notify.h>
#include "src/main/cpp/util/logging.h"
#include "src/main/native/darwin/util.h"
#include "src/main/native/unix_jni.h"
namespace blaze_jni {
static MemoryPressureLevel g_memory_pressure_level = MemoryPressureLevelNormal;
static void call_memory_pressure_callback(
dispatch_source_memorypressure_flags_t pressure_level) {
if (pressure_level == DISPATCH_MEMORYPRESSURE_NORMAL) {
BAZEL_LOG(USER) << "memory pressure normal anomaly";
g_memory_pressure_level = MemoryPressureLevelNormal;
} else if (pressure_level == DISPATCH_MEMORYPRESSURE_WARN) {
BAZEL_LOG(USER) << "memory pressure warning anomaly";
g_memory_pressure_level = MemoryPressureLevelWarning;
} else if (pressure_level == DISPATCH_MEMORYPRESSURE_CRITICAL) {
BAZEL_LOG(USER) << "memory pressure critical anomaly";
g_memory_pressure_level = MemoryPressureLevelCritical;
} else {
BAZEL_LOG(FATAL) << "unknown memory pressure level: " << pressure_level;
}
memory_pressure_callback(g_memory_pressure_level);
}
MemoryPressureLevel portable_memory_pressure() {
return g_memory_pressure_level;
}
void portable_start_memory_pressure_monitoring() {
// To test use:
// sudo memory_pressure -S -l warn
// sudo memory_pressure -S -l critical
// or use the test notifications that we register.
static dispatch_once_t once_token;
dispatch_once(&once_token, ^{
dispatch_queue_t queue = bazel::darwin::JniDispatchQueue();
dispatch_source_t source = dispatch_source_create(
DISPATCH_SOURCE_TYPE_MEMORYPRESSURE, 0,
DISPATCH_MEMORYPRESSURE_NORMAL | DISPATCH_MEMORYPRESSURE_WARN |
DISPATCH_MEMORYPRESSURE_CRITICAL,
queue);
BAZEL_CHECK_NE(source, nullptr);
dispatch_source_set_event_handler(source, ^{
dispatch_source_memorypressure_flags_t pressure_level =
dispatch_source_get_data(source);
call_memory_pressure_callback(pressure_level);
});
dispatch_resume(source);
// These are registered solely so we can test the system from end-to-end.
// Using the Apple memory_pressure requires admin access.
int testToken;
int32_t status = notify_register_dispatch(
"com.google.bazel.test.memorypressurelevel", &testToken, queue,
^(int token) {
uint64_t state;
uint32_t status = notify_get_state(token, &state);
if (status != NOTIFY_STATUS_OK) {
BAZEL_LOG(FATAL) << "notify_get_state failed: " << status;
}
dispatch_source_memorypressure_flags_t pressure_level;
switch (state) {
case 0:
pressure_level = DISPATCH_MEMORYPRESSURE_NORMAL;
break;
case 1:
pressure_level = DISPATCH_MEMORYPRESSURE_WARN;
break;
case 2:
pressure_level = DISPATCH_MEMORYPRESSURE_CRITICAL;
break;
default:
BAZEL_LOG(FATAL)
<< "unknown memory pressure level: " << state;
pressure_level = -1;
break;
}
call_memory_pressure_callback(pressure_level);
});
BAZEL_CHECK_EQ(status, NOTIFY_STATUS_OK);
});
}
} // namespace blaze_jni