LCOV - code coverage report
Current view: top level - drivers/base - map.c (source / functions) Hit Total Coverage
Test: combined.info Lines: 61 88 69.3 %
Date: 2022-03-28 16:04:14 Functions: 3 4 75.0 %
Branches: 21 40 52.5 %

           Branch data     Line data    Source code
       1                 :            : // SPDX-License-Identifier: GPL-2.0
       2                 :            : /*
       3                 :            :  *  linux/drivers/base/map.c
       4                 :            :  *
       5                 :            :  * (C) Copyright Al Viro 2002,2003
       6                 :            :  *
       7                 :            :  * NOTE: data structure needs to be changed.  It works, but for large dev_t
       8                 :            :  * it will be too slow.  It is isolated, though, so these changes will be
       9                 :            :  * local to that file.
      10                 :            :  */
      11                 :            : 
      12                 :            : #include <linux/module.h>
      13                 :            : #include <linux/slab.h>
      14                 :            : #include <linux/mutex.h>
      15                 :            : #include <linux/kdev_t.h>
      16                 :            : #include <linux/kobject.h>
      17                 :            : #include <linux/kobj_map.h>
      18                 :            : 
      19                 :            : struct kobj_map {
      20                 :            :         struct probe {
      21                 :            :                 struct probe *next;
      22                 :            :                 dev_t dev;
      23                 :            :                 unsigned long range;
      24                 :            :                 struct module *owner;
      25                 :            :                 kobj_probe_t *get;
      26                 :            :                 int (*lock)(dev_t, void *);
      27                 :            :                 void *data;
      28                 :            :         } *probes[255];
      29                 :            :         struct mutex *lock;
      30                 :            : };
      31                 :            : 
      32                 :       1638 : int kobj_map(struct kobj_map *domain, dev_t dev, unsigned long range,
      33                 :            :              struct module *module, kobj_probe_t *probe,
      34                 :            :              int (*lock)(dev_t, void *), void *data)
      35                 :            : {
      36                 :       1638 :         unsigned n = MAJOR(dev + range - 1) - MAJOR(dev) + 1;
      37                 :       1638 :         unsigned index = MAJOR(dev);
      38                 :       1638 :         unsigned i;
      39                 :       1638 :         struct probe *p;
      40                 :            : 
      41                 :       1638 :         if (n > 255)
      42                 :            :                 n = 255;
      43                 :            : 
      44                 :       1638 :         p = kmalloc_array(n, sizeof(struct probe), GFP_KERNEL);
      45         [ +  - ]:       1638 :         if (p == NULL)
      46                 :            :                 return -ENOMEM;
      47                 :            : 
      48         [ +  + ]:       3276 :         for (i = 0; i < n; i++, p++) {
      49                 :       1638 :                 p->owner = module;
      50                 :       1638 :                 p->get = probe;
      51                 :       1638 :                 p->lock = lock;
      52                 :       1638 :                 p->dev = dev;
      53                 :       1638 :                 p->range = range;
      54                 :       1638 :                 p->data = data;
      55                 :            :         }
      56                 :       1638 :         mutex_lock(domain->lock);
      57         [ +  + ]:       3276 :         for (i = 0, p -= n; i < n; i++, p++, index++) {
      58                 :       1638 :                 struct probe **s = &domain->probes[index % 255];
      59   [ +  -  -  + ]:       1638 :                 while (*s && (*s)->range < range)
      60                 :          0 :                         s = &(*s)->next;
      61                 :       1638 :                 p->next = *s;
      62                 :       1638 :                 *s = p;
      63                 :            :         }
      64                 :       1638 :         mutex_unlock(domain->lock);
      65                 :       1638 :         return 0;
      66                 :            : }
      67                 :            : 
      68                 :          0 : void kobj_unmap(struct kobj_map *domain, dev_t dev, unsigned long range)
      69                 :            : {
      70                 :          0 :         unsigned n = MAJOR(dev + range - 1) - MAJOR(dev) + 1;
      71                 :          0 :         unsigned index = MAJOR(dev);
      72                 :          0 :         unsigned i;
      73                 :          0 :         struct probe *found = NULL;
      74                 :            : 
      75                 :          0 :         if (n > 255)
      76                 :            :                 n = 255;
      77                 :            : 
      78                 :          0 :         mutex_lock(domain->lock);
      79         [ #  # ]:          0 :         for (i = 0; i < n; i++, index++) {
      80                 :          0 :                 struct probe **s;
      81         [ #  # ]:          0 :                 for (s = &domain->probes[index % 255]; *s; s = &(*s)->next) {
      82                 :          0 :                         struct probe *p = *s;
      83   [ #  #  #  # ]:          0 :                         if (p->dev == dev && p->range == range) {
      84                 :          0 :                                 *s = p->next;
      85         [ #  # ]:          0 :                                 if (!found)
      86                 :          0 :                                         found = p;
      87                 :            :                                 break;
      88                 :            :                         }
      89                 :            :                 }
      90                 :            :         }
      91                 :          0 :         mutex_unlock(domain->lock);
      92                 :          0 :         kfree(found);
      93                 :          0 : }
      94                 :            : 
      95                 :        676 : struct kobject *kobj_lookup(struct kobj_map *domain, dev_t dev, int *index)
      96                 :            : {
      97                 :        676 :         struct kobject *kobj;
      98                 :        676 :         struct probe *p;
      99                 :        676 :         unsigned long best = ~0UL;
     100                 :            : 
     101                 :        689 : retry:
     102                 :        689 :         mutex_lock(domain->lock);
     103         [ +  - ]:       2535 :         for (p = domain->probes[MAJOR(dev) % 255]; p; p = p->next) {
     104                 :       2535 :                 struct kobject *(*probe)(dev_t, int *, void *);
     105                 :       2535 :                 struct module *owner;
     106                 :       2535 :                 void *data;
     107                 :            : 
     108   [ +  +  -  + ]:       2535 :                 if (p->dev > dev || p->dev + p->range - 1 < dev)
     109                 :       1846 :                         continue;
     110         [ +  - ]:        689 :                 if (p->range - 1 >= best)
     111                 :            :                         break;
     112         [ -  + ]:        689 :                 if (!try_module_get(p->owner))
     113                 :          0 :                         continue;
     114                 :        689 :                 owner = p->owner;
     115                 :        689 :                 data = p->data;
     116                 :        689 :                 probe = p->get;
     117                 :        689 :                 best = p->range - 1;
     118                 :        689 :                 *index = dev - p->dev;
     119   [ +  +  -  + ]:        689 :                 if (p->lock && p->lock(dev, data) < 0) {
     120                 :          0 :                         module_put(owner);
     121                 :          0 :                         continue;
     122                 :            :                 }
     123                 :        689 :                 mutex_unlock(domain->lock);
     124                 :        689 :                 kobj = probe(dev, index, data);
     125                 :            :                 /* Currently ->owner protects _only_ ->probe() itself. */
     126                 :        689 :                 module_put(owner);
     127         [ +  + ]:        689 :                 if (kobj)
     128                 :        676 :                         return kobj;
     129                 :         13 :                 goto retry;
     130                 :            :         }
     131                 :          0 :         mutex_unlock(domain->lock);
     132                 :          0 :         return NULL;
     133                 :            : }
     134                 :            : 
     135                 :         26 : struct kobj_map *kobj_map_init(kobj_probe_t *base_probe, struct mutex *lock)
     136                 :            : {
     137                 :         26 :         struct kobj_map *p = kmalloc(sizeof(struct kobj_map), GFP_KERNEL);
     138                 :         26 :         struct probe *base = kzalloc(sizeof(*base), GFP_KERNEL);
     139                 :         26 :         int i;
     140                 :            : 
     141         [ -  + ]:         26 :         if ((p == NULL) || (base == NULL)) {
     142                 :          0 :                 kfree(p);
     143                 :          0 :                 kfree(base);
     144                 :          0 :                 return NULL;
     145                 :            :         }
     146                 :            : 
     147                 :         26 :         base->dev = 1;
     148                 :         26 :         base->range = ~0;
     149                 :         26 :         base->get = base_probe;
     150         [ +  + ]:       6656 :         for (i = 0; i < 255; i++)
     151                 :       6630 :                 p->probes[i] = base;
     152                 :         26 :         p->lock = lock;
     153                 :         26 :         return p;
     154                 :            : }

Generated by: LCOV version 1.14