hash.c 2.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. /*
  2. * Some portions derived from code covered by the following notice:
  3. *
  4. * Copyright (c) 2010-2013 Intel Corporation. All rights reserved.
  5. * All rights reserved.
  6. *
  7. * Redistribution and use in source and binary forms, with or without
  8. * modification, are permitted provided that the following conditions
  9. * are met:
  10. *
  11. * * Redistributions of source code must retain the above copyright
  12. * notice, this list of conditions and the following disclaimer.
  13. * * Redistributions in binary form must reproduce the above copyright
  14. * notice, this list of conditions and the following disclaimer in
  15. * the documentation and/or other materials provided with the
  16. * distribution.
  17. * * Neither the name of Intel Corporation nor the names of its
  18. * contributors may be used to endorse or promote products derived
  19. * from this software without specific prior written permission.
  20. *
  21. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  22. * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  23. * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  24. * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  25. * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  26. * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  27. * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  28. * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  29. * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  30. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  31. * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  32. */
  33. #include <linux/hash.h>
  34. #include <linux/init.h>
  35. #include <asm/processor.h>
  36. #include <asm/cpufeature.h>
  37. #include <asm/hash.h>
  38. static inline u32 crc32_u32(u32 crc, u32 val)
  39. {
  40. #ifdef CONFIG_AS_CRC32
  41. asm ("crc32l %1,%0\n" : "+r" (crc) : "rm" (val));
  42. #else
  43. asm (".byte 0xf2, 0x0f, 0x38, 0xf1, 0xc1" : "+a" (crc) : "c" (val));
  44. #endif
  45. return crc;
  46. }
  47. static u32 intel_crc4_2_hash(const void *data, u32 len, u32 seed)
  48. {
  49. const u32 *p32 = (const u32 *) data;
  50. u32 i, tmp = 0;
  51. for (i = 0; i < len / 4; i++)
  52. seed = crc32_u32(seed, *p32++);
  53. switch (len & 3) {
  54. case 3:
  55. tmp |= *((const u8 *) p32 + 2) << 16;
  56. /* fallthrough */
  57. case 2:
  58. tmp |= *((const u8 *) p32 + 1) << 8;
  59. /* fallthrough */
  60. case 1:
  61. tmp |= *((const u8 *) p32);
  62. seed = crc32_u32(seed, tmp);
  63. break;
  64. }
  65. return seed;
  66. }
  67. static u32 intel_crc4_2_hash2(const u32 *data, u32 len, u32 seed)
  68. {
  69. const u32 *p32 = (const u32 *) data;
  70. u32 i;
  71. for (i = 0; i < len; i++)
  72. seed = crc32_u32(seed, *p32++);
  73. return seed;
  74. }
  75. void __init setup_arch_fast_hash(struct fast_hash_ops *ops)
  76. {
  77. if (cpu_has_xmm4_2) {
  78. ops->hash = intel_crc4_2_hash;
  79. ops->hash2 = intel_crc4_2_hash2;
  80. }
  81. }