com.nvidia.spark.rapids.HostAlloc.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of rapids-4-spark_2.12 Show documentation
Show all versions of rapids-4-spark_2.12 Show documentation
Creates the distribution package of the RAPIDS plugin for Apache Spark
/*
* Copyright (c) 2023, NVIDIA CORPORATION.
*
* 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.
*/
package com.nvidia.spark.rapids
import ai.rapids.cudf.{DefaultHostMemoryAllocator, HostMemoryAllocator, HostMemoryBuffer, MemoryBuffer, PinnedMemoryPool}
import com.nvidia.spark.rapids.jni.{CpuRetryOOM, RmmSpark}
import org.apache.spark.internal.Logging
private class HostAlloc(nonPinnedLimit: Long) extends HostMemoryAllocator with Logging {
private var currentNonPinnedAllocated: Long = 0L
private val pinnedLimit: Long = PinnedMemoryPool.getTotalPoolSizeBytes
// For now we are going to assume that we are the only ones calling into the pinned pool
// That is not really true, but should be okay.
private var currentPinnedAllocated: Long = 0L
private val isUnlimited = nonPinnedLimit < 0
private val isPinnedOnly = nonPinnedLimit == 0
/**
* A callback class so we know when a non-pinned host buffer was released
*/
private class OnCloseCallback(ptr: Long, amount: Long) extends MemoryBuffer.EventHandler {
override def onClosed(refCount: Int): Unit = {
if (refCount == 0) {
releaseNonPinned(ptr, amount)
}
}
}
/**
* A callback so we know when a pinned host buffer was released.
*/
private class OnPinnedCloseCallback(ptr: Long, amount: Long) extends MemoryBuffer.EventHandler {
override def onClosed(refCount: Int): Unit = {
if (refCount == 0) {
releasePinned(ptr, amount)
}
}
}
private def releasePinned(ptr: Long, amount: Long): Unit = {
synchronized {
currentPinnedAllocated -= amount
}
RmmSpark.cpuDeallocate(ptr, amount)
}
private def releaseNonPinned(ptr: Long, amount: Long): Unit = {
synchronized {
currentNonPinnedAllocated -= amount
}
RmmSpark.cpuDeallocate(ptr, amount)
}
private def tryAllocPinned(amount: Long): Option[HostMemoryBuffer] = {
val ret = Option(PinnedMemoryPool.tryAllocate(amount))
ret.foreach { b =>
synchronized {
currentPinnedAllocated += amount
}
HostAlloc.addEventHandler(b, new OnPinnedCloseCallback(b.getAddress, amount))
}
ret
}
private def tryAllocNonPinned(amount: Long): Option[HostMemoryBuffer] = {
val ret = if (isUnlimited) {
synchronized {
currentNonPinnedAllocated += amount
}
Some(HostMemoryBuffer.allocate(amount, false))
} else {
synchronized {
if ((currentNonPinnedAllocated + amount) <= nonPinnedLimit) {
currentNonPinnedAllocated += amount
Some(HostMemoryBuffer.allocate(amount, false))
} else {
None
}
}
}
ret.foreach { b =>
HostAlloc.addEventHandler(b, new OnCloseCallback(b.getAddress, amount))
}
ret
}
private def canNeverSucceed(amount: Long, preferPinned: Boolean): Boolean = synchronized {
val pinnedFailed = (isPinnedOnly || preferPinned) && (amount > pinnedLimit)
val nonPinnedFailed = isPinnedOnly || (amount > nonPinnedLimit)
!isUnlimited && pinnedFailed && nonPinnedFailed
}
private def checkSize(amount: Long, preferPinned: Boolean): Unit = synchronized {
if (canNeverSucceed(amount, preferPinned)) {
throw new IllegalArgumentException(s"The amount requested $amount is larger than the " +
s"maximum pool size ${math.max(pinnedLimit, nonPinnedLimit)}")
}
}
private def spillAndCheckRetry(allocSize: Long, retryCount: Long): Boolean = {
// check arguments for good measure
require(allocSize >= 0,
s"spillAndCheckRetry invoked with invalid allocSize $allocSize")
require(retryCount >= 0,
s"spillAndCheckRetry invoked with invalid retryCount $retryCount")
val store = RapidsBufferCatalog.getHostStorage
val storeSize = store.currentSize
val storeSpillableSize = store.currentSpillableSize
val totalSize: Long = synchronized {
currentPinnedAllocated + currentNonPinnedAllocated
}
val attemptMsg = if (retryCount > 0) {
s"Attempt $retryCount"
} else {
"First attempt"
}
logInfo(s"Host allocation of $allocSize bytes failed, host store has " +
s"$storeSize total and $storeSpillableSize spillable bytes. $attemptMsg.")
if (storeSpillableSize == 0) {
logWarning(s"Host store exhausted, unable to allocate $allocSize bytes. " +
s"Total host allocated is $totalSize bytes.")
false
} else {
val targetSize = Math.max(storeSpillableSize - allocSize, 0)
logDebug(s"Targeting host store size of $targetSize bytes")
// We could not make it work so try and spill enough to make it work
val maybeAmountSpilled =
RapidsBufferCatalog.synchronousSpill(RapidsBufferCatalog.getHostStorage, targetSize)
maybeAmountSpilled.foreach { amountSpilled =>
logInfo(s"Spilled $amountSpilled bytes from the host store")
}
true
}
}
private def tryAllocInternal(amount: Long,
preferPinned: Boolean,
blocking: Boolean): (Option[HostMemoryBuffer], Boolean) = {
var retryCount = 0L
var ret = Option.empty[HostMemoryBuffer]
var shouldRetry = false
var shouldRetryInternal = true
val isRecursive = RmmSpark.preCpuAlloc(amount, blocking)
var allocAttemptFinishedWithoutException = false
try {
do {
val firstPass = if (preferPinned) {
tryAllocPinned(amount)
} else {
tryAllocNonPinned(amount)
}
ret = firstPass.orElse {
if (preferPinned) {
tryAllocNonPinned(amount)
} else {
tryAllocPinned(amount)
}
}
if (ret.isEmpty) {
// We could not make it work so try and spill enough to make it work
shouldRetryInternal = spillAndCheckRetry(amount, retryCount)
if (shouldRetryInternal) {
retryCount += 1
}
}
} while(ret.isEmpty && shouldRetryInternal && retryCount < 10)
allocAttemptFinishedWithoutException = true
} finally {
if (ret.isDefined) {
RmmSpark.postCpuAllocSuccess(ret.get.getAddress, amount, blocking, isRecursive)
} else {
// shouldRetry should indicate if spill did anything for us and we should try again.
shouldRetry = RmmSpark.postCpuAllocFailed(allocAttemptFinishedWithoutException,
blocking, isRecursive)
}
}
(ret, shouldRetry)
}
def tryAlloc(amount: Long, preferPinned: Boolean = true): Option[HostMemoryBuffer] = {
if (canNeverSucceed(amount, preferPinned)) {
return None
}
var shouldRetry = true
var ret = Option.empty[HostMemoryBuffer]
while (shouldRetry) {
val (r, sr) = tryAllocInternal(amount, preferPinned, blocking = false)
ret = r
shouldRetry = sr
}
ret
}
def alloc(amount: Long, preferPinned: Boolean = true): HostMemoryBuffer = {
checkSize(amount, preferPinned)
var ret = Option.empty[HostMemoryBuffer]
var count = 0
while (ret.isEmpty && count < 1000) {
val (r, _) = tryAllocInternal(amount, preferPinned, blocking = true)
ret = r
count += 1
}
if (ret.isEmpty) {
// This can happen if someone broke the rules and not all host memory is
// spillable when doing an allocation, like if not all of the code has
// been updated yet.
throw new CpuRetryOOM("Could not complete allocation after 1000 retries")
}
ret.get
}
override def allocate(amount: Long, preferPinned: Boolean): HostMemoryBuffer =
alloc(amount, preferPinned)
override def allocate(amount: Long): HostMemoryBuffer =
alloc(amount)
}
/**
* A new API for host memory allocation. This can be used to limit the amount of host memory.
*/
object HostAlloc {
private var singleton: HostAlloc = new HostAlloc(-1)
private def getSingleton: HostAlloc = synchronized {
singleton
}
def initialize(nonPinnedLimit: Long): Unit = synchronized {
singleton = new HostAlloc(nonPinnedLimit)
DefaultHostMemoryAllocator.set(singleton)
}
def tryAlloc(amount: Long, preferPinned: Boolean = true): Option[HostMemoryBuffer] = {
getSingleton.tryAlloc(amount, preferPinned)
}
def alloc(amount: Long, preferPinned: Boolean = true): HostMemoryBuffer = {
getSingleton.alloc(amount, preferPinned)
}
def addEventHandler(buff: HostMemoryBuffer,
handler: MemoryBuffer.EventHandler): HostMemoryBuffer = {
buff.synchronized {
val previous = Option(buff.getEventHandler)
val handlerToSet = previous.map { p =>
MultiEventHandler(p, handler)
}.getOrElse {
handler
}
buff.setEventHandler(handlerToSet)
buff
}
}
private def removeEventHandlerFrom(
multiEventHandler: MultiEventHandler,
handler: MemoryBuffer.EventHandler): MemoryBuffer.EventHandler = {
if (multiEventHandler.a == handler) {
multiEventHandler.b
} else if (multiEventHandler.b == handler) {
multiEventHandler.a
} else multiEventHandler.a match {
case oldA: MultiEventHandler =>
// From how the MultiEventHandler is set up we know that b cannot be one
val newA = removeEventHandlerFrom(oldA, handler)
MultiEventHandler(newA, multiEventHandler.b)
case _ =>
multiEventHandler
}
}
def removeEventHandler(buff: HostMemoryBuffer,
handler: MemoryBuffer.EventHandler): HostMemoryBuffer = {
buff.synchronized {
val newHandler = buff.getEventHandler match {
case multi: MultiEventHandler =>
removeEventHandlerFrom(multi, handler)
case other =>
if (other == handler) null else other
}
buff.setEventHandler(newHandler)
buff
}
}
private def findEventHandlerInternal[K](handler: MemoryBuffer.EventHandler,
eh: PartialFunction[MemoryBuffer.EventHandler, K]): Option[K] = handler match {
case multi: MultiEventHandler =>
findEventHandlerInternal(multi.a, eh)
.orElse(findEventHandlerInternal(multi.b, eh))
case other =>
eh.lift(other)
}
def findEventHandler[K](buff: HostMemoryBuffer)(
eh: PartialFunction[MemoryBuffer.EventHandler, K]): Option[K] = {
buff.synchronized {
findEventHandlerInternal(buff.getEventHandler, eh)
}
}
private case class MultiEventHandler(a: MemoryBuffer.EventHandler,
b: MemoryBuffer.EventHandler)
extends MemoryBuffer.EventHandler {
override def onClosed(i: Int): Unit = {
var t: Option[Throwable] = None
try {
a.onClosed(i)
} catch {
case e: Throwable =>
t = Some(e)
}
try {
b.onClosed(i)
} catch {
case e: Throwable =>
t match {
case Some(previousError) =>
previousError.addSuppressed(e)
case None =>
t = Some(e)
}
}
t.foreach { error =>
throw error
}
}
}
}