/* SPDX-License-Identifier: Apache-2.0 * Copyright 2021-2022 Blender Foundation */ #ifdef WITH_METAL # include "device/metal/util.h" # include "device/metal/device_impl.h" # include "util/md5.h" # include "util/path.h" # include "util/string.h" # include "util/time.h" # include # include # include CCL_NAMESPACE_BEGIN MetalGPUVendor MetalInfo::get_vendor_from_device_name(string const &device_name) { if (device_name.find("Intel") != string::npos) { return METAL_GPU_INTEL; } else if (device_name.find("AMD") != string::npos) { return METAL_GPU_AMD; } else if (device_name.find("Apple") != string::npos) { return METAL_GPU_APPLE; } return METAL_GPU_UNKNOWN; } vector> const &MetalInfo::get_usable_devices() { static vector> usable_devices; static bool already_enumerated = false; if (already_enumerated) { return usable_devices; } metal_printf("Usable Metal devices:\n"); for (id device in MTLCopyAllDevices()) { const char *device_name = [device.name UTF8String]; MetalGPUVendor vendor = get_vendor_from_device_name(device_name); bool usable = false; if (@available(macos 12.2, *)) { usable |= (vendor == METAL_GPU_APPLE); } if (@available(macos 12.3, *)) { usable |= (vendor == METAL_GPU_AMD); } if (usable) { metal_printf("- %s\n", device_name); [device retain]; usable_devices.push_back(device); } else { metal_printf(" (skipping \"%s\")\n", device_name); } } if (usable_devices.empty()) { metal_printf(" No usable Metal devices found\n"); } already_enumerated = true; return usable_devices; } id MetalBufferPool::get_buffer(id device, id command_buffer, NSUInteger length, MTLResourceOptions options, const void *pointer, Stats &stats) { id buffer; MTLStorageMode storageMode = MTLStorageMode((options & MTLResourceStorageModeMask) >> MTLResourceStorageModeShift); MTLCPUCacheMode cpuCacheMode = MTLCPUCacheMode((options & MTLResourceCPUCacheModeMask) >> MTLResourceCPUCacheModeShift); buffer_mutex.lock(); for (auto entry = buffer_free_list.begin(); entry != buffer_free_list.end(); entry++) { MetalBufferListEntry bufferEntry = *entry; /* Check if buffer matches size and storage mode and is old enough to reuse */ if (bufferEntry.buffer.length == length && storageMode == bufferEntry.buffer.storageMode && cpuCacheMode == bufferEntry.buffer.cpuCacheMode) { buffer = bufferEntry.buffer; buffer_free_list.erase(entry); bufferEntry.command_buffer = command_buffer; buffer_in_use_list.push_back(bufferEntry); buffer_mutex.unlock(); /* Copy over data */ if (pointer) { memcpy(buffer.contents, pointer, length); if (bufferEntry.buffer.storageMode == MTLStorageModeManaged) { [buffer didModifyRange:NSMakeRange(0, length)]; } } return buffer; } } // NSLog(@"Creating buffer of length %lu (%lu)", length, frameCount); if (pointer) { buffer = [device newBufferWithBytes:pointer length:length options:options]; } else { buffer = [device newBufferWithLength:length options:options]; } MetalBufferListEntry buffer_entry(buffer, command_buffer); stats.mem_alloc(buffer.allocatedSize); total_temp_mem_size += buffer.allocatedSize; buffer_in_use_list.push_back(buffer_entry); buffer_mutex.unlock(); return buffer; } void MetalBufferPool::process_command_buffer_completion(id command_buffer) { assert(command_buffer); thread_scoped_lock lock(buffer_mutex); /* Release all buffers that have not been recently reused back into the free pool */ for (auto entry = buffer_in_use_list.begin(); entry != buffer_in_use_list.end();) { MetalBufferListEntry buffer_entry = *entry; if (buffer_entry.command_buffer == command_buffer) { entry = buffer_in_use_list.erase(entry); buffer_entry.command_buffer = nil; buffer_free_list.push_back(buffer_entry); } else { entry++; } } } MetalBufferPool::~MetalBufferPool() { thread_scoped_lock lock(buffer_mutex); /* Release all buffers that have not been recently reused */ for (auto entry = buffer_free_list.begin(); entry != buffer_free_list.end();) { MetalBufferListEntry buffer_entry = *entry; id buffer = buffer_entry.buffer; // NSLog(@"Releasing buffer of length %lu (%lu) (%lu outstanding)", buffer.length, frameCount, // bufferFreeList.size()); total_temp_mem_size -= buffer.allocatedSize; [buffer release]; entry = buffer_free_list.erase(entry); } } CCL_NAMESPACE_END #endif /* WITH_METAL */