1/* SPDX-License-Identifier: GPL-2.0 */ 2/* 3 * Floating proportions with flexible aging period 4 * 5 * Copyright (C) 2011, SUSE, Jan Kara <jack@suse.cz> 6 */ 7 8#ifndef _LINUX_FLEX_PROPORTIONS_H 9#define _LINUX_FLEX_PROPORTIONS_H 10 11#include <linux/percpu_counter.h> 12#include <linux/spinlock.h> 13#include <linux/seqlock.h> 14#include <linux/gfp.h> 15 16/* 17 * When maximum proportion of some event type is specified, this is the 18 * precision with which we allow limitting. Note that this creates an upper 19 * bound on the number of events per period like 20 * ULLONG_MAX >> FPROP_FRAC_SHIFT. 21 */ 22#define FPROP_FRAC_SHIFT 10 23#define FPROP_FRAC_BASE (1UL << FPROP_FRAC_SHIFT) 24 25/* 26 * ---- Global proportion definitions ---- 27 */ 28struct fprop_global { 29 /* Number of events in the current period */ 30 struct percpu_counter events; 31 /* Current period */ 32 unsigned int period; 33 /* Synchronization with period transitions */ 34 seqcount_t sequence; 35}; 36 37int fprop_global_init(struct fprop_global *p, gfp_t gfp); 38void fprop_global_destroy(struct fprop_global *p); 39bool fprop_new_period(struct fprop_global *p, int periods); 40 41/* 42 * ---- PERCPU ---- 43 */ 44struct fprop_local_percpu { 45 /* the local events counter */ 46 struct percpu_counter events; 47 /* Period in which we last updated events */ 48 unsigned int period; 49 raw_spinlock_t lock; /* Protect period and numerator */ 50}; 51 52int fprop_local_init_percpu(struct fprop_local_percpu *pl, gfp_t gfp); 53void fprop_local_destroy_percpu(struct fprop_local_percpu *pl); 54void __fprop_add_percpu(struct fprop_global *p, struct fprop_local_percpu *pl, 55 long nr); 56void __fprop_add_percpu_max(struct fprop_global *p, 57 struct fprop_local_percpu *pl, int max_frac, long nr); 58void fprop_fraction_percpu(struct fprop_global *p, 59 struct fprop_local_percpu *pl, unsigned long *numerator, 60 unsigned long *denominator); 61 62static inline 63void fprop_inc_percpu(struct fprop_global *p, struct fprop_local_percpu *pl) 64{ 65 unsigned long flags; 66 67 local_irq_save(flags); 68 __fprop_add_percpu(p, pl, 1); 69 local_irq_restore(flags); 70} 71 72#endif 73