2023-01-24 09:55:28 +08:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								# Synchronization primitives
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								## Overview of CUDA's synchronization methods
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								The CUDA programming model provides 3 abstractions:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								*  hierarchical parallelism -- that is, parallel threads 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  grouped into hierarchical units such as blocks and clusters;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								*  shared memory, through which parallel threads that are 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  in the same hierarchical unit can communicate; and
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								*  synchronization methods for threads. 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								These abstractions help developers extract
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								both fine-grained and coarse-grained parallelism,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								by making it possible for them to subdivide problems
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								into independent components,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								and to insert synchronization at appropriate points.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Over the years CUDA has introduced several synchronization primitives
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								that operate at different levels of the hierarchy.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								These include
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								*  [thread block - level ](https://docs.nvidia.com/cuda/cuda-c-programming-guide/index.html#synchronization-functions ) synchronization (e.g., `__syncthreads()` ); 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								*  [warp-level ](https://developer.nvidia.com/blog/using-cuda-warp-level-primitives/ ) synchronization (e.g., `__syncwarp()` ); and 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								*  [thread-level ](https://docs.nvidia.com/cuda/cuda-c-programming-guide/#memory-fence-functions ) fence operations. 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								As an extension to this, starting with the Hopper architecture, CUDA added the following improvements:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								*  [thread block clusters ](https://docs.nvidia.com/cuda/cuda-c-programming-guide/index.html#thread-block-clusters ) -- 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  a new level in the thread hierarchy representing
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  a group of thread blocks that can coordinate and share data;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								*  synchronization instructions for a thread block cluster and threads within a cluster scope. 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								## CUTLASS's abstractions for Hopper features
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								CUTLASS now includes abstractions
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								for the following features introduced in Hopper.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								1.  Thread block cluster - level synchronization and query 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								   [APIs ](/include/cute/arch/cluster_sm90.hpp )
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								2.  Abstractions for new 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								   [barrier instructions ](/include/cutlass/arch/barrier.h )
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								   which help with efficient synchronization
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								   of threads within a thread block cluster.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								### Asynchronous pipelines
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								In order to write a performant GEMM Kernel,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								software pipelining is critical to hide the latency of global memory loads.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								(Please refer to the
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[Efficient GEMM ](/media/docs/efficient_gemm.md#pipelining ) document.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Different threads or groups of threads
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								may have different roles in the pipeline.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Some are "producers" that load data or perform computations
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								to satisfy other threads' input data dependencies.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								The same or different threads may be "consumers"
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								that do other work with those input data dependencies,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								once they are satisfied.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Starting with the Hopper architecture,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								the presence of hardware-accelerated synchronization instructions
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								make it possible for "producer" and "consumer" threads
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								to communicate with each other efficiently
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								about their data dependencies.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Implementing a persistent GEMM algorithm calls for managing
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								dozens of different kinds of asynchronously executing operations
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								that synchronize using multiple barriers organized as a circular list.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								This complexity is too much for human programmers to manage by hand.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								As a result, we have developed
							 
						 
					
						
							
								
									
										
										
										
											2023-10-19 00:55:46 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[asynchronous Pipeline classes ](/include/cutlass/pipeline/ ).
							 
						 
					
						
							
								
									
										
										
										
											2023-01-24 09:55:28 +08:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								These classes help developers orchestrate a pipeline
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								of asynchronous producer and consumer threads,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								without needing to worry about lower-level hardware details.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								These classes serve a similar function as the various
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[pipeline abstractions ](https://nvidia.github.io/libcudacxx/extended_api/synchronization_primitives/pipeline.html )
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								in libcu++.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								#### Pipeline methods 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								##### Producer acquire 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								The `producer_acquire`  method is to be used by asynchronous producer threads
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								before issuing other instructions associated with a particular pipeline stage
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								(e.g., copy or write).
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								This is a blocking instruction
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								which blocks further execution of consumer threads
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								unless the particular stage waiting to be acquired
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								is released by a consumer.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								We say that a pipeline at its start is "empty" if producer threads are free to produce and do not need to wait for a consumer release -- that is, if an acquire operation is expected to succeed.  If the pipeline at its start is empty, then we can either skip performing producer acquire operations during the first pass through the pipeline stages, or use the `make_producer_start_state`  method.  The latter ensures that the acquire operation will succeed at the start of a pipeline.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								##### Producer commit
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								The `producer_commit`  method is to be issued by asynchronous producer threads
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								after the instructions associated with a particular stage
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								(e.g., shared memory writes) have completed,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								in order to notify the waiting asynchronous consumer threads.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								This is a nonblocking instruction.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								This API may result in a No-Op in some cases,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								if the producer instructions also update the barrier stage associated automatically
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								(e.g., TMA_based producer threads using the  `PipelineTmaAsync `  class).
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								##### Consumer wait
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								The `consumer_wait`  method is to be used by consumer threads
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								before consuming data from a particular pipeline stage
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								which is expected to be produced by producer threads.  
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								This is a blocking instruction.  That is,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								until the producer threads have committed to a particular stage,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								this instruction is expected to block further execution of consumer threads.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								##### Consumer release
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								The `consumer_release`  method is to be used by consumer threads
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								to signal waiting producer threads that they have finished consuming data
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								associated with a particular stage of the pipeline.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								This is a nonblocking instruction.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								#### Pipeline example
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								```c++
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								// 4-stage Pipeline
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								static constexpr int NumStages = 4;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								using MainloopPipeline = typename cutlass::PipelineAsync< NumStages > ;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								using PipelineState = typename cutlass::PipelineState< NumStages > ;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								// 2 producer threads and 1 consumer thread 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								typename MainloopPipeline::Params params;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								params.producer_arv_count = 2;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								params.consumer_arv_count = 1;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								MainloopPipeline pipeline(shared_storage.storage, params);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								// Producer threads
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								if (thread_idx == 0 or thread_idx == 1) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  PipelineState smem_pipe_write = cutlass::make_producer_start_state< MainloopPipeline > ();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  for ( ; iter > 0; --iter) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    pipeline.producer_acquire(smem_pipe_write);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // Producer ops
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // If any memory operations are involved, then we also need
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // to guarantee that writes are completed and visible to consumer(s).
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-04-15 11:19:34 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    pipeline.producer_commit(smem_pipe_write);
							 
						 
					
						
							
								
									
										
										
										
											2023-01-24 09:55:28 +08:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    ++smem_pipe_write;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								else if (thread_idx == 2) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  PipelineState smem_pipe_read;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  for (; iter > 0; --iter) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    pipeline.consumer_wait(smem_pipe_read);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // Consumer ops
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    pipeline.consumer_release(smem_pipe_read);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ++smem_pipe_read;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								```
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								In this example, we create an instance of the asynchronous pipeline class `PipelineSync` ,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								and then synchronize among 3 asynchronously executing threads:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								2 producer threads and 1 consumer thread.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Please note that this is a basic example.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								There are different versions possible,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								depending on what the producer and consumer threads are doing.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Please refer to our [unit tests ](/test/unit/pipeline )
							 
						 
					
						
							
								
									
										
										
										
											2023-11-03 12:01:25 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								and the other [pipeline classes ](/include/cutlass/pipeline/pipeline.hpp )
							 
						 
					
						
							
								
									
										
										
										
											2023-01-24 09:55:28 +08:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								for more details.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# Copyright
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-01-17 03:37:22 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Copyright (c) 2023 - 2024 NVIDIA CORPORATION &  AFFILIATES. All rights reserved.
							 
						 
					
						
							
								
									
										
										
										
											2023-01-24 09:55:28 +08:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								SPDX-License-Identifier: BSD-3-Clause
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								```
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  Redistribution and use in source and binary forms, with or without
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  modification, are permitted provided that the following conditions are met:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  1.  Redistributions of source code must retain the above copyright notice, this
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  list of conditions and the following disclaimer.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  2.  Redistributions in binary form must reproduce the above copyright notice,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  this list of conditions and the following disclaimer in the documentation
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  and/or other materials provided with the distribution.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  3.  Neither the name of the copyright holder nor the names of its
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  contributors may be used to endorse or promote products derived from
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  this software without specific prior written permission.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								```