From 0649cbfc8d6debf8740d3d12f8723e58a517f1cd Mon Sep 17 00:00:00 2001 From: Vladislav Bolkhovitin Date: Mon, 28 Mar 2011 11:31:49 +0000 Subject: [PATCH] SGV cache doc converted to LinuxDoc git-svn-id: http://svn.code.sf.net/p/scst/svn/trunk@3316 d57e44dd-8a1f-0410-8b47-8ef2f437770f --- doc/Makefile | 2 - doc/{sgv_cache.txt => sgv_cache.sgml} | 209 +++++++++++++++++++------- 2 files changed, 155 insertions(+), 56 deletions(-) rename doc/{sgv_cache.txt => sgv_cache.sgml} (60%) diff --git a/doc/Makefile b/doc/Makefile index 26994cd3f..44257779d 100644 --- a/doc/Makefile +++ b/doc/Makefile @@ -36,9 +36,7 @@ rtf: $(RTFS) $(COMMAND)rtf $(<) clean: - @mv sgv_cache.txt sgv_cache.tx rm -f *.txt *.html *.tex *.dvi *.ps *.pdf *.info *.lyx *.rtf - @mv sgv_cache.tx sgv_cache.txt extraclean: clean rm -f *.orig *.rej diff --git a/doc/sgv_cache.txt b/doc/sgv_cache.sgml similarity index 60% rename from doc/sgv_cache.txt rename to doc/sgv_cache.sgml index 6814dbc51..4d6240031 100644 --- a/doc/sgv_cache.txt +++ b/doc/sgv_cache.sgml @@ -1,9 +1,22 @@ - SCST SGV CACHE. + - PROGRAMMING INTERFACE DESCRIPTION. +
- For SCST version 1.0.2 + +SCST SGV cache description + + + Vladislav Bolkhovitin + + +Version 2.1.0 + + + +Introduction + +

SCST SGV cache is a memory management subsystem in SCST. One can call it a "memory pool", but Linux kernel already have a mempool interface, which serves different purposes. SGV cache provides to SCST core, target @@ -13,50 +26,65 @@ facility, when it doesn't free to the system each vector, which is not used anymore, but keeps it for a while (possibly indefinitely) to let it be reused by the next consecutive command. This allows to: - - Reduce commands processing latencies and, hence, improve performance; + - - Make commands processing latencies predictable, which is essential + Reduce commands processing latencies and, hence, improve performance; + + Make commands processing latencies predictable, which is essential for RT applications. + + The freed SG vectors are kept by the SGV cache either for some (possibly indefinite) time, or, optionally, until the system needs more memory and asks to free some using the set_shrinker() interface. Also the SGV cache allows to: - - Cluster pages together. "Cluster" means merging adjacent pages in a + + + Cluster pages together. "Cluster" means merging adjacent pages in a single SG entry. It allows to have less SG entries in the resulting SG vector, hence improve performance handling it as well as allow to work with bigger buffers on hardware with limited SG capabilities. - - Set custom page allocator functions. For instance, scst_user device + Set custom page allocator functions. For instance, scst_user device handler uses this facility to eliminate unneeded mapping/unmapping of user space pages and avoid unneeded IOCTL calls for buffers allocations. In fileio_tgt application, which uses a regular malloc() function to allocate data buffers, this facility allows ~30% less CPU load and considerable performance increase. - - Prevent each initiator or all initiators altogether to allocate too + Prevent each initiator or all initiators altogether to allocate too much memory and DoS the target. Consider 10 initiators, which can have access to 10 devices each. Any of them can queue up to 64 commands, each can transfer up to 1MB of data. So, all of them in a peak can allocate up to 10*10*64 = ~6.5GB of memory for data buffers. This amount must be limited somehow and the SGV cache performs this function. + + + Implementation + +

From implementation POV the SGV cache is a simple extension of the kmem cache. It can work in 2 modes: -1. With fixed size buffers. + -2. With a set of power 2 size buffers. In this mode each SGV cache + With fixed size buffers. + + With a set of power 2 size buffers. In this mode each SGV cache (struct sgv_pool) has SGV_POOL_ELEMENTS (11 currently) of kmem caches. Each of those kmem caches keeps SGV cache objects (struct sgv_pool_obj) corresponding to SG vectors with size of order X pages. For instance, -request to allocate 4 pages will be served from kmem cache[2], since the +request to allocate 4 pages will be served from kmem cache[2&rsqb, since the order of the of number of requested pages is 2. If later request to allocate 11KB comes, the same SG vector with 4 pages will be reused (see below). This mode is in average allows less memory overhead comparing with the fixed size buffers mode. + + Consider how the SGV cache works in the set of buffers mode. When a request to allocate new SG vector comes, sgv_pool_alloc() via sgv_get_obj() checks if there is already a cached vector with that @@ -82,153 +110,226 @@ Freed cached sgv_pool_obj objects are actually freed to the system either by the purge work, which is scheduled once in 60 seconds, or in sgv_shrink() called by system, when it's asking for memory. + Interface - Interface. + sgv_pool *sgv_pool_create() -struct sgv_pool *sgv_pool_create(const char *name, +

+ +struct sgv_pool *sgv_pool_create( + const char *name, enum sgv_clustering_types clustered, int single_alloc_pages, bool shared, int purge_interval) + This function creates and initializes an SGV cache. It has the following arguments: - - name - the name of the SGV cache + - - clustered - sets type of the pages clustering. The type can be: + - * sgv_full_clustering - free merging of pages at any place in - the SG is allowed. This mode usually provides the best merging - rate. + + + 0, then the SGV cache will work in the fixed size buffers mode. In this case single_alloc_pages sets the size of each buffer in pages. - - shared - sets if the SGV cache can be shared between devices or not. + + Returns the resulting SGV cache or NULL in case of any error. + void sgv_pool_del() -void sgv_pool_del(struct sgv_pool *pool) +

+ +void sgv_pool_del( + struct sgv_pool *pool) + This function deletes the corresponding SGV cache. If the cache is shared, it will decrease its reference counter. If the reference counter reaches 0, the cache will be destroyed. + void sgv_pool_flush() -void sgv_pool_flush(struct sgv_pool *pool) +

+ +void sgv_pool_flush( + struct sgv_pool *pool) + This function flushes, i.e. frees, all the cached entries in the SGV cache. + void sgv_pool_set_allocator() -void sgv_pool_set_allocator(struct sgv_pool *pool, +

+ +void sgv_pool_set_allocator( + struct sgv_pool *pool, struct page *(*alloc_pages_fn)(struct scatterlist *sg, gfp_t gfp, void *priv), void (*free_pages_fn)(struct scatterlist *sg, int sg_count, void *priv)); + This function allows to set for the SGV cache a custom pages allocator. For instance, scst_user uses such function to supply to the cache mapped from user space pages. -alloc_pages_fn() has the following parameters: + - - gfp - the allocation GFP flags + This function should return the allocated page or NULL, if no page was allocated. -free_pages_fn() has the following parameters: + - - sg_count - number of SG entries in the sg + + + struct scatterlist *sgv_pool_alloc() + +

+ +struct scatterlist *sgv_pool_alloc( + struct sgv_pool *pool, + unsigned int size, + gfp_t gfp_mask, + int flags, + int *count, + struct sgv_pool_obj **sgv, + struct scst_mem_lim *mem_lim, + void *priv) + This function allocates an SG vector from the SGV cache. It has the following parameters: - - pool - the cache to alloc from + - - size - size of the resulting SG vector in bytes + - * SGV_POOL_NO_ALLOC_ON_CACHE_MISS - don't do an allocation on a + - - sgv - the resulting SGV object. It should be used to free the + + This function returns pointer to the resulting SG vector or NULL in case of any error. + void sgv_pool_free() -void sgv_pool_free(struct sgv_pool_obj *sgv, struct scst_mem_lim *mem_lim) +

+ +void sgv_pool_free( + struct sgv_pool_obj *sgv, + struct scst_mem_lim *mem_lim) + This function frees previously allocated SG vector, referenced by SGV cache object sgv. + void *sgv_get_priv(struct sgv_pool_obj *sgv) -void *sgv_get_priv(struct sgv_pool_obj *sgv) +

+ +void *sgv_get_priv( + struct sgv_pool_obj *sgv) + This function allows to get the allocation private data for this SGV cache object sgv. The private data are set by sgv_pool_alloc(). + void scst_init_mem_lim() -void scst_init_mem_lim(struct scst_mem_lim *mem_lim) +

+ +void scst_init_mem_lim( + struct scst_mem_lim *mem_lim) + This function initializes memory limits structure mem_lim according to the current system configuration. This structure should be latter used to track and limit allocated by one or more SGV caches memory. - Runtime information and statistics. + Runtime information and statistics. +

Runtime information and statistics is available in /sys/kernel/scst_tgt/sgv. +