cache.c
上传用户:lgb322
上传日期:2013-02-24
资源大小:30529k
文件大小:8k
源码类别:

嵌入式Linux

开发平台:

Unix_Linux

  1. /*
  2.  *
  3.  *
  4.  *  Copyright (C) 2000 Stelias Computing, Inc.
  5.  *  Copyright (C) 2000 Red Hat, Inc.
  6.  *
  7.  *
  8.  */
  9. #define __NO_VERSION__
  10. #include <linux/module.h>
  11. #include <stdarg.h>
  12. #include <asm/bitops.h>
  13. #include <asm/uaccess.h>
  14. #include <asm/system.h>
  15. #include <linux/errno.h>
  16. #include <linux/fs.h>
  17. #include <linux/ext2_fs.h>
  18. #include <linux/slab.h>
  19. #include <linux/vmalloc.h>
  20. #include <linux/sched.h>
  21. #include <linux/stat.h>
  22. #include <linux/string.h>
  23. #include <linux/locks.h>
  24. #include <linux/blkdev.h>
  25. #include <linux/init.h>
  26. #include <linux/intermezzo_fs.h>
  27. #include <linux/intermezzo_upcall.h>
  28. #include <linux/intermezzo_psdev.h>
  29. /*
  30.    This file contains the routines associated with managing a
  31.    cache of files for InterMezzo.  These caches have two reqs:
  32.    - need to be found fast so they are hashed by the device, 
  33.      with an attempt to have collision chains of length 1.
  34.    The methods for the cache are set up in methods.
  35. */
  36. /* the intent of this hash is to have collision chains of length 1 */
  37. #define CACHES_BITS 8
  38. #define CACHES_SIZE (1 << CACHES_BITS)
  39. #define CACHES_MASK CACHES_SIZE - 1
  40. static struct list_head presto_caches[CACHES_SIZE];
  41. static inline int presto_cache_hash(kdev_t dev)
  42. {
  43.         return (CACHES_MASK) & ((0x000F & (dev)) + ((0x0F00 & (dev)) >>8));
  44. }
  45. inline void presto_cache_add(struct presto_cache *cache, kdev_t dev)
  46. {
  47.         list_add(&cache->cache_chain,
  48.                  &presto_caches[presto_cache_hash(dev)]);
  49.         cache->cache_dev = dev;
  50. }
  51. inline void presto_init_cache_hash(void)
  52. {
  53.         int i;
  54.         for ( i = 0; i < CACHES_SIZE; i++ ) {
  55.                 INIT_LIST_HEAD(&presto_caches[i]);
  56.         }
  57. }
  58. /* map a device to a cache */
  59. struct presto_cache *presto_find_cache(kdev_t dev)
  60. {
  61.         struct presto_cache *cache;
  62.         struct list_head *lh, *tmp;
  63.         lh = tmp = &(presto_caches[presto_cache_hash(dev)]);
  64.         while ( (tmp = lh->next) != lh ) {
  65.                 cache = list_entry(tmp, struct presto_cache, cache_chain);
  66.                 if ( cache->cache_dev == dev ) {
  67.                         return cache;
  68.                 }
  69.         }
  70.         return NULL;
  71. }
  72. /* map an inode to a cache */
  73. struct presto_cache *presto_get_cache(struct inode *inode)
  74. {
  75.         struct presto_cache *cache;
  76.         /* find the correct presto_cache here, based on the device */
  77.         cache = presto_find_cache(inode->i_dev);
  78.         if ( !cache ) {
  79.                 printk("WARNING: no presto cache for dev %x, ino %ldn",
  80.                        inode->i_dev, inode->i_ino);
  81.                 EXIT;
  82.                 return NULL;
  83.         }
  84.         return cache;
  85. }
  86. /* list cache mount points for ioctl's or /proc/fs/intermezzo/mounts */
  87. int presto_sprint_mounts(char *buf, int buflen, int minor)
  88. {
  89.         int len = 0;
  90.         int i;
  91.         struct list_head *head, *tmp;
  92.         struct presto_cache *cache;
  93.         buf[0] = '';
  94.         for (i=0 ; i<CACHES_SIZE ; i++) {
  95.                 head = tmp = &presto_caches[i];
  96.                 while ( (tmp = tmp->next) != head ) {
  97.                         cache = list_entry(tmp, struct presto_cache,
  98.                                             cache_chain);
  99.                         if ( !cache->cache_root_fileset || !cache->cache_mtpt)
  100.                                 continue;
  101.                         if ((minor != -1) &&
  102.                             (cache->cache_psdev->uc_minor != minor))
  103.                                 continue;
  104.                         if ( strlen(cache->cache_root_fileset) +
  105.                              strlen(cache->cache_mtpt) + 
  106.                              strlen(cache->cache_psdev->uc_devname) +
  107.                              4 > buflen - len)
  108.                                 break;
  109.                         len += sprintf(buf + len, "%s %s %sn",
  110.                                        cache->cache_root_fileset,
  111.                                        cache->cache_mtpt,
  112.                                        cache->cache_psdev->uc_devname);
  113.                 }
  114.         }
  115.         buf[buflen-1] = '';
  116.         CDEBUG(D_SUPER, "%sn", buf);
  117.         return len;
  118. }
  119. #ifdef CONFIG_KREINT
  120. /* get mount point by volname
  121.        Arthur Ma, 2000.12.25
  122.  */
  123. int presto_get_mount (char *buf, int buflen, char *volname)
  124. {
  125.         int i;
  126.         struct list_head *head, *tmp;
  127.         struct presto_cache *cache = NULL;
  128.         char *path = "";
  129.         buf[0] = '';
  130.         for (i=0 ; i<CACHES_SIZE ; i++) {
  131.                 head = tmp = &presto_caches[i];
  132.                 while ( (tmp = tmp->next) != head ) {
  133.                         cache = list_entry(tmp, struct presto_cache,
  134.                                             cache_chain);
  135.                         if ( !cache->cache_root_fileset || !cache->cache_mtpt)
  136.                                 continue;
  137.                         if ( strcmp(cache->cache_root_fileset, volname) == 0)
  138.                                 break;
  139.                 }
  140.         }
  141.         if (cache != NULL)
  142.                 path = cache->cache_mtpt;
  143.         strncpy (buf, path, buflen);
  144.         return strlen (buf);
  145. }
  146. #endif
  147. /* another debugging routine: check fs is InterMezzo fs */
  148. int presto_ispresto(struct inode *inode)
  149. {
  150.         struct presto_cache *cache;
  151.         if ( !inode )
  152.                 return 0;
  153.         cache = presto_get_cache(inode);
  154.         if ( !cache )
  155.                 return 0;
  156.         return (inode->i_dev == cache->cache_dev);
  157. }
  158. /* setup a cache structure when we need one */
  159. struct presto_cache *presto_init_cache(void)
  160. {
  161.         struct presto_cache *cache;
  162.         /* make a presto_cache structure for the hash */
  163.         PRESTO_ALLOC(cache, struct presto_cache *, sizeof(struct presto_cache));
  164.         if ( cache ) {
  165.                 memset(cache, 0, sizeof(struct presto_cache));
  166.                 INIT_LIST_HEAD(&cache->cache_chain);
  167.                 INIT_LIST_HEAD(&cache->cache_fset_list);
  168.         }
  169. cache->cache_lock = SPIN_LOCK_UNLOCKED;
  170. cache->cache_reserved = 0; 
  171.         return cache;
  172. }
  173. /* free a cache structure and all of the memory it is pointing to */
  174. inline void presto_free_cache(struct presto_cache *cache)
  175. {
  176.         if (!cache)
  177.                 return;
  178.         list_del(&cache->cache_chain);
  179.         if (cache->cache_mtpt)
  180.                 PRESTO_FREE(cache->cache_mtpt, strlen(cache->cache_mtpt) + 1);
  181.         if (cache->cache_type)
  182.                 PRESTO_FREE(cache->cache_type, strlen(cache->cache_type) + 1);
  183.         if (cache->cache_root_fileset)
  184.                 PRESTO_FREE(cache->cache_root_fileset, strlen(cache->cache_root_fileset) + 1);
  185.         PRESTO_FREE(cache, sizeof(struct presto_cache));
  186. }
  187. int presto_reserve_space(struct presto_cache *cache, loff_t req)
  188. {
  189.         struct filter_fs *filter; 
  190.         loff_t avail; 
  191. struct super_block *sb = cache->cache_sb;
  192.         filter = cache->cache_filter;
  193. if (!filter ) {
  194. EXIT;
  195. return 0; 
  196. }
  197. if (!filter->o_trops ) {
  198. EXIT;
  199. return 0; 
  200. }
  201. if (!filter->o_trops->tr_avail ) {
  202. EXIT;
  203. return 0; 
  204. }
  205.         avail = filter->o_trops->tr_avail(cache, sb); 
  206.         CDEBUG(D_SUPER, "ESC::%ld +++> %ld n", (long) cache->cache_reserved,
  207.          (long) (cache->cache_reserved + req)); 
  208.         CDEBUG(D_SUPER, "ESC::Avail::%ld n", (long) avail);
  209. spin_lock(&cache->cache_lock);
  210.         if (req + cache->cache_reserved > avail) {
  211. spin_unlock(&cache->cache_lock);
  212.                 EXIT;
  213.                 return -ENOSPC;
  214.         }
  215. cache->cache_reserved += req; 
  216. spin_unlock(&cache->cache_lock);
  217.         return 0;
  218. }
  219. void presto_release_space(struct presto_cache *cache, loff_t req)
  220. {
  221.         CDEBUG(D_SUPER, "ESC::%ld ---> %ld n", (long) cache->cache_reserved,
  222.          (long) (cache->cache_reserved - req)); 
  223. spin_lock(&cache->cache_lock);
  224. cache->cache_reserved -= req; 
  225. spin_unlock(&cache->cache_lock);
  226. }