디시인사이드 갤러리

갤러리 이슈박스, 최근방문 갤러리

갤러리 본문 영역

러스트가 개떡같은 EU

나르시갤로그로 이동합니다. 2025.07.23 14:59:03
조회 37 추천 0 댓글 0

러스트로는 이런거 못 짠다.

ㅋㅋㅋ

/* -*- Mode: C; indent-tabs-mode: nil; c-basic-offset: 2; tab-width: 2 -*- */

/*

  c-hash-map.c

  Copyright (C) 2021-2024 Hodong Kim, All rights reserved.

  Unauthorized copying of this software, via any medium is strictly prohibited.

  Proprietary and confidential.

  Written by Hodong Kim <hodong@nimfsoft.art>

 */

#if defined(__FreeBSD__) || defined(__OpenBSD__)

  #include <sys/endian.h>

#else

  #define _DEFAULT_SOURCE

  #include <endian.h>

  #include <sys/random.h>

#endif

#include "c-hash-map.h"

#include <stdlib.h>

#include <string.h>

#include "c-array.h"

#include "c-mem.h"

#include "c-list.h"


/*****************************************************************************/


typedef struct _CEntry CEntry;

struct _CEntry {

  CEntry* next;

  CEntry* prev;

  void*   key;

  void*   value;

};


static CEntry* c_entry_new (void* key, void* value)

{

  CEntry* entry = c_malloc (sizeof (CEntry));

  entry->key    = key;

  entry->value  = value;

  return entry;

}


static void c_entry_free (CEntry*   entry,

                          CFreeFunc key_free_func,

                          CFreeFunc value_free_func)

{

  if (key_free_func)

    key_free_func (entry->key);


  if (value_free_func)

    value_free_func (entry->value);


  free (entry);

}


/*****************************************************************************/


uint32_t c_murmur3_32 (const uint8_t* key, size_t len, uint32_t seed)

{

  uint32_t h = seed;

  uint32_t k;


  for (size_t i = len >> 2; i; i--)

  {

    k = htole32 (*(uint32_t*) key);

    key += sizeof (uint32_t);

    k *= 0xcc9e2d51;

    k = (k << 15) | (k >> 17);

    k *= 0x1b873593;

    h ^= k;

    h = (h << 13) | (h >> 19);

    h = h * 5 + 0xe6546b64;

  }


  k = 0;


  for (size_t i = len & 3; i; i--)

  {

    k <<= 8;

    k |= key[i - 1];

  }


  k *= 0xcc9e2d51;

  k = (k << 15) | (k >> 17);

  k *= 0x1b873593;

  h ^= k;


  h ^= len;

  h ^= h >> 16;

  h *= 0x85ebca6b;

  h ^= h >> 13;

  h *= 0xc2b2ae35;

  h ^= h >> 16;


  return h;

}


uintptr_t c_str_hash (const void* str, uint32_t seed)

{

  return c_murmur3_32 (str, strlen (str), seed);

}


/*****************************************************************************/


// RAND_MAX:      0x7fffffffU  This is a prime number

// UINT32_MAX:    0xffffffffU

#if defined(__FreeBSD__) || defined(__OpenBSD__) || defined(__linux__)

  #define PRIME     0xfffffffbU /* for arc4random_uniform() or getrandom() */

#else

  #define PRIME     0x7fffffffU /* for rand() */

#endif


typedef struct _CBucket CBucket;

struct _CBucket {

  unsigned random;      // 1 <= k <= prime - 1

  unsigned entries_len; // entries_len = 2 * m * (m - 1)

  unsigned m;           // entries_len = 2 * m * (m - 1)

  unsigned b;           // rehash: b = lists[i]->len, insert: b = b + 1

  CEntry** entries;

};


// constant

#define DYNFECT_C  1 // m = (1 + c) * C_MAX (count, 4) in rehash()

#define DYNFECT_S  1 // buckets_len = S(M) = DYNFECT_S * dynfect->m


typedef struct _CDynfect CDynfect;

struct _CDynfect {

  unsigned   random;      /* 1 <= k <= prime - 1 */

  unsigned   buckets_len; /* buckets_len = S(M) = DYNFECT_S * dynfect->m */

  CHashFunc  key_hash_func;

  CEqualFunc key_equal_func;

  CFreeFunc  key_free_func;

  CFreeFunc  value_free_func;

  CList*     list;

  CBucket*   buckets;

  unsigned   op_count; /* count++ when add, remove */

  unsigned   m;        /* m = (1 + c) * C_MAX (count, 4) in rehash() */

};


static unsigned c_dynfect_gen_random ()

{

#if !defined(__FreeBSD__) && !defined(__OpenBSD__) && !defined(__linux__)

  #if RAND_MAX != 0x7fffffffU

    #error "Debug this source code"

  #endif

  static unsigned seed;


  if (!seed)

  {

    /* This is insecure */

    seed = time (NULL);

    srand (seed);

  }

#endif

  /* h(x) = (kx mod p) mod s; 1 <= k <= prime - 1 */

#if defined(__FreeBSD__) || defined(__OpenBSD__)

  return arc4random_uniform (PRIME - 1) + 1; /* uint32_t */

#elif defined __linux__

  uint32_t buf;

  if (getrandom (&buf, sizeof (uint32_t), 0) != -1)

    return buf % (PRIME - 1) + 1;

  else

    return rand() % (PRIME - 1) + 1; /* int */

#else

  return rand() % (PRIME - 1) + 1; /* int */

#endif

}


uintptr_t c_ptr_hash (const void* p, uint32_t seed)

{

  return (seed * (uintptr_t) p) % PRIME;

}


bool c_ptr_equal (const void* a, const void* b)

{

  return a == b;

}


static bool c_dynfect_condition (CDynfect* dynfect)

{

  unsigned sum = 0;

  // 32 * M * M / S(M) + 4 * M

  // S(M) is buckets_len

  // buckets_len is DYNFECT_S * dynfect->m

  unsigned bound = 32 * dynfect->m / DYNFECT_S + 4 * dynfect->m;


  for (unsigned i = 0; i < dynfect->buckets_len; i++)

  {

    sum = sum + dynfect->buckets[i].m;


    if (sum > bound)

      return false;

  }


  return true;

}


static bool c_dynfect_inject (CDynfect* dynfect, CArray* list, CBucket* bucket)

{

  bucket->random = c_dynfect_gen_random ();


  for (unsigned i = 0; i < list->len; i++)

  {

    CEntry*  entry = list->data[i];

    unsigned index = dynfect->key_hash_func (entry->key, bucket->random) %

                     (unsigned long) bucket->entries_len;

    if (bucket->entries[index])

      return false;


    bucket->entries[index] = entry;

  }


  return true;

}


static bool c_dynfect_contains (CDynfect* dynfect, const void* key)

{

  unsigned i = dynfect->key_hash_func (key, dynfect->random) % (unsigned long) dynfect->buckets_len;

  CBucket* bucket = &dynfect->buckets[i];


  if (bucket->entries_len > 0)

  {

    int pos = dynfect->key_hash_func (key, bucket->random) % (unsigned long) bucket->entries_len;

    CEntry* entry = bucket->entries[pos];


    if (entry && dynfect->key_equal_func (entry->key, key))

      return true;

  }


  return false;

}


static void c_dynfect_rehash (CDynfect* dynfect, CEntry* new_entry)

{

  CArray*  list;

  CArray** lists;


  list = c_array_new (NULL, true);


  for (unsigned i = 0; i < dynfect->buckets_len; i++)

  {

    CBucket* bucket = &dynfect->buckets[i];

    for (unsigned j = 0; j < bucket->entries_len; j++)

      if (bucket->entries[j])

        c_array_add (list, bucket->entries[j]);


    free (bucket->entries);

  }


  if (new_entry)

    c_array_add (list, new_entry);


  dynfect->op_count = list->len;

  dynfect->m = (1 + DYNFECT_C) * C_MAX (dynfect->op_count, 4);

  dynfect->buckets_len = DYNFECT_S * dynfect->m;


  dynfect->buckets = c_realloc (dynfect->buckets,

                                dynfect->buckets_len * sizeof (CBucket));

  memset (dynfect->buckets, 0, dynfect->buckets_len * sizeof (CBucket));

  lists = c_calloc (dynfect->buckets_len, sizeof (CArray*));


  do {

    dynfect->random = c_dynfect_gen_random ();


    for (unsigned i = 0; i < dynfect->buckets_len; i++)

      if (lists[i] && lists[i]->len)

        c_array_clear (lists[i]);


    for (unsigned i = 0; i < list->len; i++)

    {

      CEntry* entry = list->data[i];

      unsigned pos = dynfect->key_hash_func (entry->key, dynfect->random) % (unsigned long) dynfect->buckets_len;


      if (!lists[pos])

        lists[pos] = c_array_new (NULL, true);


      c_array_add (lists[pos], entry);

    }


    for (unsigned i = 0; i < dynfect->buckets_len; i++)

    {

      CBucket* bucket = &dynfect->buckets[i];

      memset (bucket, 0, sizeof (CBucket));

      bucket->b = lists[i] ? lists[i]->len : 0;

      bucket->m = 2 * bucket->b;

      bucket->entries_len = 2 * bucket->m * (bucket->m - 1);

    }

  } while (!c_dynfect_condition (dynfect));


  c_array_free (list);


  for (unsigned i = 0; i < dynfect->buckets_len; i++)

  {

    CBucket* bucket = &dynfect->buckets[i];


    if (lists[i] && (lists[i]->len > 0))

    {

      bucket->entries = c_calloc (bucket->entries_len, sizeof (CEntry*));


      do {

        memset (bucket->entries, 0, bucket->entries_len * sizeof (CEntry*));

      } while (!c_dynfect_inject (dynfect, lists[i], bucket));

    }


    if (lists[i])

      c_array_free (lists[i]);

  }


  free (lists);

}


static CDynfect* c_dynfect_new (CHashFunc  key_hash_func,

                                CEqualFunc key_equal_func,

                                CFreeFunc  key_free_func,

                                CFreeFunc  value_free_func,

                                CList*     list)

{

  CDynfect* dynfect = c_malloc (sizeof (CDynfect));


  dynfect->key_hash_func   = key_hash_func;

  dynfect->key_equal_func  = key_equal_func;

  dynfect->key_free_func   = key_free_func;

  dynfect->value_free_func = value_free_func;

  dynfect->list            = list;

  dynfect->op_count        = 0;

  dynfect->buckets_len     = 0;

  dynfect->buckets         = NULL;

  dynfect->m               = 0;

  c_dynfect_rehash (dynfect, NULL);


  return dynfect;

}


static bool

c_dynfect_insert (CDynfect* dynfect, void* key, void* value, CEntry* new_entry)

{

  if (new_entry)

    key = new_entry->key;


  unsigned index  = dynfect->key_hash_func (key, dynfect->random) %

                    (unsigned long) dynfect->buckets_len;

  CBucket* bucket = &dynfect->buckets[index];

  unsigned pos    = 0; // avoid maybe-uninitialized


  if (bucket->entries_len > 0)

  {

    pos = dynfect->key_hash_func (key, bucket->random) %

          (unsigned long) bucket->entries_len;

    CEntry* entry = bucket->entries[pos];


    if (entry)

    {

      if (dynfect->key_equal_func (entry->key, key))

      {

        if (dynfect->key_free_func)

          dynfect->key_free_func (entry->key);


        if (dynfect->value_free_func)

          dynfect->value_free_func (entry->value);


        entry->key   = key;

        entry->value = value;


        return false;

      }

    }

  }


  if (!new_entry)

  {

    new_entry = c_entry_new (key, value);

    c_list_insert_end (dynfect->list, (CNode*) new_entry);

  }


  dynfect->op_count++;


  if (dynfect->op_count > dynfect->m)

  {

    c_dynfect_rehash (dynfect, new_entry);


    return true;

  }


  bucket->b++;


  if (bucket->b <= bucket->m)

  {

    /* bucket->entries_len > 0 is true, so 'pos' NOT maybe-uninitialized */

    if (!bucket->entries[pos])

    {

      bucket->entries[pos] = new_entry;

    }

    else

    {

      CArray* list;


      list = c_array_new (NULL, true);


      for (unsigned i = 0; i < bucket->entries_len; i++)

        if (bucket->entries[i])

          c_array_add (list, bucket->entries[i]);


      c_array_add (list, new_entry);


      do {

        if (bucket->entries == NULL)

          bucket->entries = c_calloc (bucket->entries_len, sizeof (CEntry*));

        else

          memset (bucket->entries,

                  0, bucket->entries_len * sizeof (CEntry*));

      } while (!c_dynfect_inject (dynfect, list, bucket));


      c_array_free (list);

    }

  }

  else

  {

    if (c_dynfect_condition (dynfect))

    {

      CArray* list;


      list = c_array_new (NULL, true);


      if (bucket->entries)

        for (unsigned j = 0; j < bucket->entries_len; j++)

          if (bucket->entries[j])

            c_array_add (list, bucket->entries[j]);


      c_array_add (list, new_entry);


      bucket->m = 2 * C_MAX (bucket->m, 1);

      bucket->entries_len = 2 * (bucket->m) * (bucket->m - 1);

      bucket->entries = c_realloc (bucket->entries,

                                   bucket->entries_len * sizeof (CEntry*));

      do {

        memset (bucket->entries, 0, bucket->entries_len * sizeof (CEntry*));

      } while (!c_dynfect_inject (dynfect, list, bucket));


      c_array_free (list);

    }

    else

    {

      c_dynfect_rehash (dynfect, new_entry);

    }

  }


  return true;

}


static bool c_dynfect_remove (CDynfect* dynfect, const void* key)

{

  dynfect->op_count++;


  unsigned i = dynfect->key_hash_func (key, dynfect->random) %

                                       (unsigned long) dynfect->buckets_len;

  CBucket* bucket = &dynfect->buckets[i];


  if (bucket->entries_len < 1)

    return false;


  int pos = dynfect->key_hash_func (key, bucket->random) %

                                    (unsigned long) bucket->entries_len;

  CEntry* entry = bucket->entries[pos];


  if (entry == NULL)

    return false;


  c_list_remove_link (dynfect->list, (CNode*) entry);

  bucket->entries[pos] = NULL;


  if (dynfect->op_count >= dynfect->m)

    c_dynfect_rehash (dynfect, NULL);


  c_entry_free (entry, dynfect->key_free_func, dynfect->value_free_func);


  return true;

}


static void* c_dynfect_lookup (CDynfect* dynfect, const void* key)

{

  unsigned i = dynfect->key_hash_func (key, dynfect->random) % (unsigned long) dynfect->buckets_len;

  CBucket* bucket = &dynfect->buckets[i];


  if (bucket->entries_len > 0)

  {

    int pos = dynfect->key_hash_func (key, bucket->random) % (unsigned long) bucket->entries_len;

    CEntry* entry = bucket->entries[pos];


    if (entry && dynfect->key_equal_func (entry->key, key))

      return entry->value;

  }


  return NULL;

}


static void c_dynfect_free (CDynfect* dynfect)

{

  for (unsigned i = 0; i < dynfect->buckets_len; i++)

  {

    CBucket* bucket = &dynfect->buckets[i];

    free (bucket->entries);

  }


  free (dynfect->buckets);

  free (dynfect);

}



추천 비추천

0

고정닉 0

0

댓글 영역

전체 댓글 0
본문 보기

하단 갤러리 리스트 영역

왼쪽 컨텐츠 영역

갤러리 리스트 영역

갤러리 리스트
번호 제목 글쓴이 작성일 조회 추천
설문 모태 솔로도 구제해 줄 것 같은 연애 고수 스타는? 운영자 25/07/21 - -
2874105 짱깨들도 2찢명 쿠폰으로 한국세금 달달하거이 빨아먹는즁 ㅋㅅㅋ ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 27 0
2874104 물리 배우다 프로그래밍으로 넘어가서 편했던점 [2] ㅆㅇㅆ찡갤로그로 이동합니다. 07.24 69 0
2874103 나님만큼 최신 기술에 거부감 없이 잘쓰는 인재는 드뭄 ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 38 0
2874102 여기서 조언바래봤자 딴데 가는게 나음 [2] ㅆㅇㅆ찡갤로그로 이동합니다. 07.24 54 0
2874101 나님이 이번에 실험중인 항목들 ㅇㅅㅇ ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 29 0
2874100 2ㅌㅊ 까지는 올해마무리 할지 모르겠넹 ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 33 0
2874099 늒비 간곡한 조언 부탁드립니다 [1] 프갤러(203.10) 07.24 42 0
2874098 요즘 사람들 이를 안 닦나보군요 루도그담당(211.184) 07.24 46 0
2874097 중간 일정까지 고려하면 올해 간당하겠고 ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 25 0
2874096 단순 계산으로 75면.. 흠.. ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 26 0
2874095 내일 출근 안합니다 ㅋ 루도그담당(211.184) 07.24 40 0
2874093 지방은 경력 10년도 연봉 5천 미만 수두룩하네 프갤러(218.238) 07.24 45 0
2874092 웹 영상 drm 걸린건 왠만해선 복호화 불가능함? [8] ㅇㅇ(110.10) 07.24 88 0
2874091 프로그래밍 솔직히 한 5~6년전 코드보면 기억 안나서 느끼는거지만 [1] ㅆㅇㅆ(124.216) 07.24 58 0
2874090 나님 통찰력 ㄱㅆㅅㅌㅊ.. ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 29 0
2874089 언젠간 이 문서를 완성하는게 내 개인적 목표임 [2] ㅆㅇㅆ(124.216) 07.24 52 0
2874088 저기 밑에 WPF 예제 추천해줌 ㅇㅇ [4] ㅆㅇㅆ(124.216) 07.24 69 0
2874087 C#에서 고급 문법이랄게 뭐가 있을까 [1] 루도그담당(211.184) 07.24 53 0
2874086 C# WPF에 윈폼이면 그렇게 어려울거 없이 1년투자하면 될거같은데 [5] ㅆㅇㅆ(124.216) 07.24 85 0
2874085 내가 느끼는건 프로그래밍하다보면 자신만의 기술 개념의 계층이 필요함 [2] ㅆㅇㅆ(124.216) 07.24 61 0
2874083 나는 항상 프로그래밍 메타 따라갈려고 체크중인데 내가 느끼는게 ㅆㅇㅆ(124.216) 07.24 45 0
2874082 마음아프고 애정결핍애들 애반게리온 필수시청 [4] 프갤러(183.101) 07.24 45 0
2874081 날씨 미쳤네 진짜 루도그담당(211.184) 07.24 37 0
2874080 공수처 검찰청 경찰청 국제수사 과학수사 포랜식수사 기무사 국정원 존재이유 뒷통수한방(1.213) 07.24 24 0
2874079 봇찌 css로 만들어봄 [5] ㅇㅇ(125.205) 07.24 63 1
2874078 근데 권도형 미국 갔잖아 그래서 뭐 언급없는거 아닌가? [3] ㅆㅇㅆ(124.216) 07.24 50 0
2874076 님들 루나코인 폭락 사건 알음? [1] 프갤러(222.114) 07.24 53 0
2874075 오 대기업들 슬슬 시작하려는듯 ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 47 0
2874074 인정에서 발전이 시작되는법 ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 32 0
2874073 나님 회복탄력성 실험즁❤+ ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 35 0
2874072 나님 25억 인증❤+ ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 43 0
2874071 개인적으로 느끼는거지만 결국 프로그래밍은 어떤 글쓰기 스타일이냐가 [2] ㅆㅇㅆ(124.216) 07.24 80 2
2874070 길가다 찍은 일본의 묘한 선거포스터 [9] 아스카영원히사랑해갤로그로 이동합니다. 07.24 97 0
2874069 최악무능 친중좌파 2찢명 입구컷! ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 40 0
2874068 신입, 주니어 취준생들 이력서 첨삭해줌 프갤러(221.148) 07.24 39 0
2874067 우낏낏끼! 우키킼! [1] 통암기원숭이(211.235) 07.24 52 0
2874066 ❤✨☀⭐⚡☘⛩나님 시작합니당⛩☘⚡⭐☀✨❤ ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 28 0
2874065 파이어 베이스 쓰는게 나을까? 프갤러(39.118) 07.24 33 0
2874064 MSA가 꼭 크기는 클 필요 없긴함 ㅆㅇㅆ(124.216) 07.24 37 0
2874063 30대 모솔아다 지잡 비전공 백수 무직 [2] 어린이노무현갤로그로 이동합니다. 07.24 71 0
2874061 msa 개인프로젝트는 뭐 어케하는거냐 [2] 밀우갤로그로 이동합니다. 07.24 60 0
2874060 스킬은 바뀐다 하지만 프갤러(211.234) 07.24 35 0
2874059 ❤✨☀⭐⚡☘⛩나님 시작합니당⛩☘⚡⭐☀✨❤ ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 28 0
2874058 [2찢명] 자녀특혜비리의혹 최휘영 제2의 조국사태 터질 조짐 ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 52 0
2874057 공수처 검찰청 경찰청 국제수사 과학수사 포랜식수사 기무사 국정원 존재이유 뒷통수한방(1.213) 07.24 27 0
2874055 이갤엔 마음이 아픈 애들이 참 많구낭.. ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 46 0
2874054 회사에서 권고사직만 허락해주면 바로나갈껀데.. [2] ㅇㅇ(211.235) 07.24 66 0
2874053 양자역학은 세상이 확률적이라고 하는데... ㅇㅇ(183.101) 07.24 46 0
2874052 학원들 대세 타는거 왜이리 빠르노 ㅇㅇ(211.235) 07.24 88 0
2874051 ❤✨☀⭐⛩⚡☘♥+나님 시작합니당♥+☘⚡⛩⭐☀✨❤ ♥팬티스타킹냥덩♥갤로그로 이동합니다. 07.24 36 0
뉴스 '청담국제고등학교 2' 이은샘-장성윤의 대립 모멘트, 숨 막히는 긴장감 유발 디시트렌드 07.25
갤러리 내부 검색
제목+내용게시물 정렬 옵션

오른쪽 컨텐츠 영역

실시간 베스트

1/8

뉴스

디시미디어

디시이슈

1/2