Path: blob/master/arch/m68k/include/asm/bitops_mm.h
10820 views
#ifndef _M68K_BITOPS_H1#define _M68K_BITOPS_H2/*3* Copyright 1992, Linus Torvalds.4*5* This file is subject to the terms and conditions of the GNU General Public6* License. See the file COPYING in the main directory of this archive7* for more details.8*/910#ifndef _LINUX_BITOPS_H11#error only <linux/bitops.h> can be included directly12#endif1314#include <linux/compiler.h>1516/*17* Require 68020 or better.18*19* They use the standard big-endian m680x0 bit ordering.20*/2122#define test_and_set_bit(nr,vaddr) \23(__builtin_constant_p(nr) ? \24__constant_test_and_set_bit(nr, vaddr) : \25__generic_test_and_set_bit(nr, vaddr))2627#define __test_and_set_bit(nr,vaddr) test_and_set_bit(nr,vaddr)2829static inline int __constant_test_and_set_bit(int nr, unsigned long *vaddr)30{31char *p = (char *)vaddr + (nr ^ 31) / 8;32char retval;3334__asm__ __volatile__ ("bset %2,%1; sne %0"35: "=d" (retval), "+m" (*p)36: "di" (nr & 7));3738return retval;39}4041static inline int __generic_test_and_set_bit(int nr, unsigned long *vaddr)42{43char retval;4445__asm__ __volatile__ ("bfset %2{%1:#1}; sne %0"46: "=d" (retval) : "d" (nr^31), "o" (*vaddr) : "memory");4748return retval;49}5051#define set_bit(nr,vaddr) \52(__builtin_constant_p(nr) ? \53__constant_set_bit(nr, vaddr) : \54__generic_set_bit(nr, vaddr))5556#define __set_bit(nr,vaddr) set_bit(nr,vaddr)5758static inline void __constant_set_bit(int nr, volatile unsigned long *vaddr)59{60char *p = (char *)vaddr + (nr ^ 31) / 8;61__asm__ __volatile__ ("bset %1,%0"62: "+m" (*p) : "di" (nr & 7));63}6465static inline void __generic_set_bit(int nr, volatile unsigned long *vaddr)66{67__asm__ __volatile__ ("bfset %1{%0:#1}"68: : "d" (nr^31), "o" (*vaddr) : "memory");69}7071#define test_and_clear_bit(nr,vaddr) \72(__builtin_constant_p(nr) ? \73__constant_test_and_clear_bit(nr, vaddr) : \74__generic_test_and_clear_bit(nr, vaddr))7576#define __test_and_clear_bit(nr,vaddr) test_and_clear_bit(nr,vaddr)7778static inline int __constant_test_and_clear_bit(int nr, unsigned long *vaddr)79{80char *p = (char *)vaddr + (nr ^ 31) / 8;81char retval;8283__asm__ __volatile__ ("bclr %2,%1; sne %0"84: "=d" (retval), "+m" (*p)85: "di" (nr & 7));8687return retval;88}8990static inline int __generic_test_and_clear_bit(int nr, unsigned long *vaddr)91{92char retval;9394__asm__ __volatile__ ("bfclr %2{%1:#1}; sne %0"95: "=d" (retval) : "d" (nr^31), "o" (*vaddr) : "memory");9697return retval;98}99100/*101* clear_bit() doesn't provide any barrier for the compiler.102*/103#define smp_mb__before_clear_bit() barrier()104#define smp_mb__after_clear_bit() barrier()105106#define clear_bit(nr,vaddr) \107(__builtin_constant_p(nr) ? \108__constant_clear_bit(nr, vaddr) : \109__generic_clear_bit(nr, vaddr))110#define __clear_bit(nr,vaddr) clear_bit(nr,vaddr)111112static inline void __constant_clear_bit(int nr, volatile unsigned long *vaddr)113{114char *p = (char *)vaddr + (nr ^ 31) / 8;115__asm__ __volatile__ ("bclr %1,%0"116: "+m" (*p) : "di" (nr & 7));117}118119static inline void __generic_clear_bit(int nr, volatile unsigned long *vaddr)120{121__asm__ __volatile__ ("bfclr %1{%0:#1}"122: : "d" (nr^31), "o" (*vaddr) : "memory");123}124125#define test_and_change_bit(nr,vaddr) \126(__builtin_constant_p(nr) ? \127__constant_test_and_change_bit(nr, vaddr) : \128__generic_test_and_change_bit(nr, vaddr))129130#define __test_and_change_bit(nr,vaddr) test_and_change_bit(nr,vaddr)131#define __change_bit(nr,vaddr) change_bit(nr,vaddr)132133static inline int __constant_test_and_change_bit(int nr, unsigned long *vaddr)134{135char *p = (char *)vaddr + (nr ^ 31) / 8;136char retval;137138__asm__ __volatile__ ("bchg %2,%1; sne %0"139: "=d" (retval), "+m" (*p)140: "di" (nr & 7));141142return retval;143}144145static inline int __generic_test_and_change_bit(int nr, unsigned long *vaddr)146{147char retval;148149__asm__ __volatile__ ("bfchg %2{%1:#1}; sne %0"150: "=d" (retval) : "d" (nr^31), "o" (*vaddr) : "memory");151152return retval;153}154155#define change_bit(nr,vaddr) \156(__builtin_constant_p(nr) ? \157__constant_change_bit(nr, vaddr) : \158__generic_change_bit(nr, vaddr))159160static inline void __constant_change_bit(int nr, unsigned long *vaddr)161{162char *p = (char *)vaddr + (nr ^ 31) / 8;163__asm__ __volatile__ ("bchg %1,%0"164: "+m" (*p) : "di" (nr & 7));165}166167static inline void __generic_change_bit(int nr, unsigned long *vaddr)168{169__asm__ __volatile__ ("bfchg %1{%0:#1}"170: : "d" (nr^31), "o" (*vaddr) : "memory");171}172173static inline int test_bit(int nr, const unsigned long *vaddr)174{175return (vaddr[nr >> 5] & (1UL << (nr & 31))) != 0;176}177178static inline int find_first_zero_bit(const unsigned long *vaddr,179unsigned size)180{181const unsigned long *p = vaddr;182int res = 32;183unsigned int words;184unsigned long num;185186if (!size)187return 0;188189words = (size + 31) >> 5;190while (!(num = ~*p++)) {191if (!--words)192goto out;193}194195__asm__ __volatile__ ("bfffo %1{#0,#0},%0"196: "=d" (res) : "d" (num & -num));197res ^= 31;198out:199res += ((long)p - (long)vaddr - 4) * 8;200return res < size ? res : size;201}202#define find_first_zero_bit find_first_zero_bit203204static inline int find_next_zero_bit(const unsigned long *vaddr, int size,205int offset)206{207const unsigned long *p = vaddr + (offset >> 5);208int bit = offset & 31UL, res;209210if (offset >= size)211return size;212213if (bit) {214unsigned long num = ~*p++ & (~0UL << bit);215offset -= bit;216217/* Look for zero in first longword */218__asm__ __volatile__ ("bfffo %1{#0,#0},%0"219: "=d" (res) : "d" (num & -num));220if (res < 32) {221offset += res ^ 31;222return offset < size ? offset : size;223}224offset += 32;225226if (offset >= size)227return size;228}229/* No zero yet, search remaining full bytes for a zero */230return offset + find_first_zero_bit(p, size - offset);231}232#define find_next_zero_bit find_next_zero_bit233234static inline int find_first_bit(const unsigned long *vaddr, unsigned size)235{236const unsigned long *p = vaddr;237int res = 32;238unsigned int words;239unsigned long num;240241if (!size)242return 0;243244words = (size + 31) >> 5;245while (!(num = *p++)) {246if (!--words)247goto out;248}249250__asm__ __volatile__ ("bfffo %1{#0,#0},%0"251: "=d" (res) : "d" (num & -num));252res ^= 31;253out:254res += ((long)p - (long)vaddr - 4) * 8;255return res < size ? res : size;256}257#define find_first_bit find_first_bit258259static inline int find_next_bit(const unsigned long *vaddr, int size,260int offset)261{262const unsigned long *p = vaddr + (offset >> 5);263int bit = offset & 31UL, res;264265if (offset >= size)266return size;267268if (bit) {269unsigned long num = *p++ & (~0UL << bit);270offset -= bit;271272/* Look for one in first longword */273__asm__ __volatile__ ("bfffo %1{#0,#0},%0"274: "=d" (res) : "d" (num & -num));275if (res < 32) {276offset += res ^ 31;277return offset < size ? offset : size;278}279offset += 32;280281if (offset >= size)282return size;283}284/* No one yet, search remaining full bytes for a one */285return offset + find_first_bit(p, size - offset);286}287#define find_next_bit find_next_bit288289/*290* ffz = Find First Zero in word. Undefined if no zero exists,291* so code should check against ~0UL first..292*/293static inline unsigned long ffz(unsigned long word)294{295int res;296297__asm__ __volatile__ ("bfffo %1{#0,#0},%0"298: "=d" (res) : "d" (~word & -~word));299return res ^ 31;300}301302#ifdef __KERNEL__303304/*305* ffs: find first bit set. This is defined the same way as306* the libc and compiler builtin ffs routines, therefore307* differs in spirit from the above ffz (man ffs).308*/309310static inline int ffs(int x)311{312int cnt;313314asm ("bfffo %1{#0:#0},%0" : "=d" (cnt) : "dm" (x & -x));315316return 32 - cnt;317}318#define __ffs(x) (ffs(x) - 1)319320/*321* fls: find last bit set.322*/323324static inline int fls(int x)325{326int cnt;327328asm ("bfffo %1{#0,#0},%0" : "=d" (cnt) : "dm" (x));329330return 32 - cnt;331}332333static inline int __fls(int x)334{335return fls(x) - 1;336}337338#include <asm-generic/bitops/fls64.h>339#include <asm-generic/bitops/sched.h>340#include <asm-generic/bitops/hweight.h>341#include <asm-generic/bitops/lock.h>342343/* Bitmap functions for the little endian bitmap. */344345static inline void __set_bit_le(int nr, void *addr)346{347__set_bit(nr ^ 24, addr);348}349350static inline void __clear_bit_le(int nr, void *addr)351{352__clear_bit(nr ^ 24, addr);353}354355static inline int __test_and_set_bit_le(int nr, void *addr)356{357return __test_and_set_bit(nr ^ 24, addr);358}359360static inline int test_and_set_bit_le(int nr, void *addr)361{362return test_and_set_bit(nr ^ 24, addr);363}364365static inline int __test_and_clear_bit_le(int nr, void *addr)366{367return __test_and_clear_bit(nr ^ 24, addr);368}369370static inline int test_and_clear_bit_le(int nr, void *addr)371{372return test_and_clear_bit(nr ^ 24, addr);373}374375static inline int test_bit_le(int nr, const void *vaddr)376{377const unsigned char *p = vaddr;378return (p[nr >> 3] & (1U << (nr & 7))) != 0;379}380381static inline int find_first_zero_bit_le(const void *vaddr, unsigned size)382{383const unsigned long *p = vaddr, *addr = vaddr;384int res = 0;385unsigned int words;386387if (!size)388return 0;389390words = (size >> 5) + ((size & 31) > 0);391while (*p++ == ~0UL) {392if (--words == 0)393goto out;394}395396--p;397for (res = 0; res < 32; res++)398if (!test_bit_le(res, p))399break;400out:401res += (p - addr) * 32;402return res < size ? res : size;403}404#define find_first_zero_bit_le find_first_zero_bit_le405406static inline unsigned long find_next_zero_bit_le(const void *addr,407unsigned long size, unsigned long offset)408{409const unsigned long *p = addr;410int bit = offset & 31UL, res;411412if (offset >= size)413return size;414415p += offset >> 5;416417if (bit) {418offset -= bit;419/* Look for zero in first longword */420for (res = bit; res < 32; res++)421if (!test_bit_le(res, p)) {422offset += res;423return offset < size ? offset : size;424}425p++;426offset += 32;427428if (offset >= size)429return size;430}431/* No zero yet, search remaining full bytes for a zero */432return offset + find_first_zero_bit_le(p, size - offset);433}434#define find_next_zero_bit_le find_next_zero_bit_le435436static inline int find_first_bit_le(const void *vaddr, unsigned size)437{438const unsigned long *p = vaddr, *addr = vaddr;439int res = 0;440unsigned int words;441442if (!size)443return 0;444445words = (size >> 5) + ((size & 31) > 0);446while (*p++ == 0UL) {447if (--words == 0)448goto out;449}450451--p;452for (res = 0; res < 32; res++)453if (test_bit_le(res, p))454break;455out:456res += (p - addr) * 32;457return res < size ? res : size;458}459#define find_first_bit_le find_first_bit_le460461static inline unsigned long find_next_bit_le(const void *addr,462unsigned long size, unsigned long offset)463{464const unsigned long *p = addr;465int bit = offset & 31UL, res;466467if (offset >= size)468return size;469470p += offset >> 5;471472if (bit) {473offset -= bit;474/* Look for one in first longword */475for (res = bit; res < 32; res++)476if (test_bit_le(res, p)) {477offset += res;478return offset < size ? offset : size;479}480p++;481offset += 32;482483if (offset >= size)484return size;485}486/* No set bit yet, search remaining full bytes for a set bit */487return offset + find_first_bit_le(p, size - offset);488}489#define find_next_bit_le find_next_bit_le490491/* Bitmap functions for the ext2 filesystem. */492493#define ext2_set_bit_atomic(lock, nr, addr) \494test_and_set_bit_le(nr, addr)495#define ext2_clear_bit_atomic(lock, nr, addr) \496test_and_clear_bit_le(nr, addr)497498#endif /* __KERNEL__ */499500#endif /* _M68K_BITOPS_H */501502503