UMAC - UMAC

İçinde kriptografi, bir temel alan mesaj kimlik doğrulama kodu evrensel hashingveya UMAC, bir tür mesaj doğrulama kodu (MAC), bazı gizli (rastgele) işleme göre bir karma işlevi sınıfından bir karma işlevi seçmeyi ve bunu iletiye uygulamayı hesapladı. Ortaya çıkan özet veya parmak izi daha sonra kullanılan karma işlevin kimliğini gizlemek için şifrelenir. Herhangi bir MAC ile olduğu gibi, her ikisini de aynı anda doğrulamak için kullanılabilir. veri bütünlüğü ve özgünlük bir İleti.

Belirli bir UMAC türü, genellikle sadece UMAC, içinde belirtilmiştir RFC 4418, kanıtlanabilir bir şifreleme gücüne sahiptir ve genellikle diğer MAC'lardan çok daha az hesaplama yoğunluğuna sahiptir. UMAC'ın tasarımı, 32 bit mimariler için optimize edilmiştir. SIMD SIMD ile bayt başına 1 CPU döngüsü (cpb) ve SIMD'siz 2 cpb performansı ile destek. 64 bit mimariler için optimize edilmiş yakından ilişkili bir UMAC varyantı, VMAC IETF'e taslak olarak sunulan (taslak-krovetz-vmac-01) ancak hiçbir zaman standartlaştırılmış bir RFC olmak için yeterince dikkat çekmedi.

Arka fon

Evrensel hashing

Hash fonksiyonunun, mesajları olası mesaj özetlerinin kümesi olan D'ye eşleyen bir H hash fonksiyonları sınıfından seçildiğini varsayalım. Bu sınıfın adı evrensel herhangi bir farklı mesaj çifti için en fazla | H | / | D | onları D'nin aynı üyesine eşleyen işlevler.

Bu, eğer bir saldırgan bir mesajı bir başkasıyla değiştirmek isterse ve onun bakış açısından hash fonksiyonu tamamen rastgele seçilmişse, UMAC'ın değiştirmeyi tespit etmeme olasılığının en fazla 1 / | D | olduğu anlamına gelir.

Ancak bu tanım yeterince güçlü değildir - olası mesajlar 0 ve 1 ise, D = {0,1} ve H kimlik işleminden oluşur ve değil, H evrenseldir. Ancak özet modüler ekleme ile şifrelenmiş olsa bile, saldırgan aynı anda mesajı ve özeti değiştirebilir ve alıcı farkı anlayamaz.

Kesinlikle evrensel hashing

Kullanması iyi olan bir H hash işlevi sınıfı, bir saldırganın doğru özeti tahmin etmesini zorlaştıracaktır. d sahte bir mesajın f bir mesajı yakaladıktan sonra a sindirimle c. Diğer bir deyişle,

çok küçük olması gerekir, tercihen 1 / |D|.

Bir hash işlevi sınıfı oluşturmak kolaydır. D dır-dir alan. Örneğin, eğer |D| dır-dir önemli tüm işlemler alınır modulo |D|. Mesaj a daha sonra bir nboyutlu vektör bitti D (a1, a2, ..., an). H sonra vardır |D|n+1 üyeler, her biri bir (n + 1) boyutlu vektör D (h0, h1, ..., hn). İzin verirsek

kanıtlamak için olasılık ve kombinatorik kurallarını kullanabiliriz

Tüm özetleri uygun şekilde şifrelersek (ör. Bir defalık ped ), bir saldırgan onlardan hiçbir şey öğrenemez ve aynı hash işlevi iki taraf arasındaki tüm iletişim için kullanılabilir. Bu doğru olmayabilir ECB şifreleme çünkü iki mesajın aynı hash değerini üretmesi oldukça muhtemel olabilir. Sonra bir çeşit başlatma vektörü sık sık adı verilen nonce. Ayarlamak yaygın bir uygulama haline geldi h0 = f(nonce), nerede f aynı zamanda sırdır.

Büyük miktarda bilgisayar gücüne sahip olmanın saldırgana hiç yardımcı olmadığına dikkat edin. Alıcı, kabul ettiği sahtecilik miktarını sınırlarsa (birini algıladığında uyuyarak), |D| 2 olabilir32 veya daha küçük.

Misal

Aşağıdaki C işlevi 24 bitlik bir UMAC üretir. Varsayar ki gizli 24 bitin katı, msg daha uzun değil gizli ve sonuç 24 gizli biti zaten içeriyor, ör. f (nonce). nonce'nin içinde yer almasına gerek yoktur msg.

C dil kodu (orijinal)
/ * ÇİFT: Bunun (muhtemelen uzun) RFC ile bir ilgisi yok gibi görünüyor * tanımlama. Bu muhtemelen genel UMAC konseptine bir örnektir. * 2007'deki (Nroets) kim bir örnekte 3 baytı seçiyor? * * Bunu daha iyi str tanımıyla birlikte hareket ettirmeliyiz. uni. hash yapmak * uni. karma. * /#define uchar uint8_tgeçersiz UHash24 (uchar *msg, uchar *gizli, size_t len, uchar *sonuç){  uchar r1 = 0, r2 = 0, r3 = 0, s1, s2, s3, byteCnt = 0, bitCnt, bayt;    süre (len-- > 0) {    / * Her üç bayt için yeni bir sır getir. * /    Eğer (byteCnt-- == 0) {      s1 = *gizli++;      s2 = *gizli++;      s3 = *gizli++;      byteCnt = 2;       }    bayt = *msg++;    / * Mesajın her bir baytı, sırların bir bitinin karma haline gelip gelmediğini kontrol eder. 、     *     * Sırasını 24'ün altında tutma konusunu anlamıyorum, çünkü 3 baytlık bir şeyle     * tanıma göre sadece 0-23 polinominal sırasını tutar. "Sn" kodunun aynı     * davranış, her bit için hala ÇOK iş yapıyor olsak da     */    için (uchar bitCnt = 0; bitCnt < 8; bitCnt++) {      / * Son bit, gizli bit kullanılıp kullanılmayacağını kontrol eder. * /      Eğer (bayt & 1) {        r1 ^= s1; / * (sn >> 16) & 0xff * /        r2 ^= s2; / * (sn >> 8) & 0xff * /        r3 ^= s3; / * (sn) & 0xff * /      }      bayt >>= 1; / * sonraki bit. * /      / * ve sırrı x (yani 2) ile çarparak (XOR ile) çıkarın         sırasını 24 (?!) altında tutmak için gerektiğinde polinom * /      uchar doSub = s3 & 0x80;      s3 <<= 1;      Eğer (s2 & 0x80) s3 |= 1;      s2 <<= 1;      Eğer (s1 & 0x80) s2 |= 1;      s1 <<= 1;      Eğer (doSub) {  / * 0b0001 1011 -> * /        s1 ^= 0x1B; / * x ^ 24 + x ^ 4 + x ^ 3 + x + 1 [16777243 - asal değil] * /      }    } / * mesajdaki her bit için * /  } / * mesajdaki her bayt için * /  *sonuç++ ^= r1;  *sonuç++ ^= r2;  *sonuç++ ^= r3;}
C dil kodu (revize edildi)
#define uchar uint8_t#define swap32 (x) ((x) ve 0xff) << 24 | ((x) ve 0xff00) << 8 | ((x) ve 0xff0000) >> 8 | (x) ve 0xff000000) >> 24)/ * Bu aynı şey, ancak gruplandırılmış (daha iyi montaj ve malzeme üretme).   Hala kötü ve kimse neden bu kadar evrensel olduğunu açıklamadı. * /geçersiz UHash24Ex (uchar *msg, uchar *gizli, size_t len, uchar *sonuç){  uchar bayt, okumak;  uint32_t saniye = 0, ret = 0, içerik = 0;  süre (len > 0) {    / * Bir yığın halinde üç tane okuyun. * /    içerik = 0;    değiştirmek (okumak = (len >= 3 ? 3 : len)) {      durum 2: içerik |= (uint32_t) msg[2] << 16; / * FALLTHRU * /      durum 1: içerik |= (uint32_t) msg[1] << 8;  / * FALLTHRU * /      durum 0: içerik |= (uint32_t) msg[0];    }    len -= okumak; msg += okumak;    / * Her üç bayt için yeni bir sır getir. * /    saniye = (uint32_t) gizli[2] << 16 | (uint32_t) gizli[1] << 8 | (uint32_t) gizli[0];    gizli += 3;    / * Harika kompresör. * /    için (bitCnt = 0; bitCnt < 24; bitCnt++) {      / * Kaldırılması gereken bir sabit veri bağımlılığı: çıktı bağlıdır       * ara üründe.       * CRC bayt tabloları ile gerçekten çalışmıyor. * /      Eğer (bayt & 1) {        ret ^= saniye;      }      bayt >>= 1; / * sonraki bit. * /      / * Kaydırma kaydı. * /      saniye <<= 1;      Eğer (saniye & 0x01000000)        saniye ^= 0x0100001B;      saniye &= 0x00ffffff;    } / * mesajdaki her bit için * /  } / * mesajdaki her 3 bayt için * /  sonuç[0] ^= ret & 0xff;  sonuç[1] ^= (ret >>  8) & 0xff;  sonuç[2] ^= (ret >> 16) & 0xff;}

NH ve RFC UMAC

NH

Yukarıdaki isimsiz işlevler[kaynak belirtilmeli ] son derece evrensel hash-function aile kullanımları n bir karma değeri hesaplamak için çarpılır.

NH ailesi, pratikte kabaca iki kat hızlanma anlamına gelen çarpma sayısını yarıya indirir.[1] Hız için UMAC, NH hash-function ailesini kullanır. NH, özellikle kullanım için tasarlanmıştır SIMD talimatlar ve dolayısıyla UMAC, SIMD için optimize edilmiş ilk MAC işlevidir.[2]

Aşağıdaki karma ailesi -evrensel:[2]

.

nerede

  • M mesajı bir nboyutlu vektör w-bit sözcükler (m0, m1, m2, ..., mn-1).
  • Ara anahtar K, bir n + 1boyutlu vektör w-bit sözcükler (k0, k1, k2, ..., kn). Bir sözde rasgele üretici paylaşılan bir gizli anahtardan K üretir.

Pratik olarak NH, işaretsiz tam sayılarla yapılır. Tüm çarpmalar mod 2 ^ şeklindedirw, tüm eklemeler mod 2 ^w/ 2 ve tüm girişler yarım kelimelerin bir vektörü (-bit tamsayılar). Algoritma daha sonra kullanacak çarpımlar, nerede vektördeki yarım kelimelerin sayısıdır. Bu nedenle, algoritma, girdi kelimesi başına bir çarpma "hızında" çalışır.

RFC 4418

RFC 4418 NH'yi iyi bir UMAC yapmak için sarmak için çok şey yapar. Genel UHASH ("Evrensel Karma İşlevi") rutini, karma işleminin üç katmanında da ihtiyaç duyulan yineleme sayısına (ve anahtarların toplam uzunluklarına) karşılık gelen değişken uzunlukta etiket üretir. AES tabanlı bir anahtar türetme işlevi üç anahtarlı karma için de anahtar sağlamak için kullanılır.

  • Katman 1 (1024 baytlık yığınlar -> 8 baytlık karmalar birleştirilmiş) hızlı olduğu için NH kullanır.
  • Katman 2, ana modül aritmetiği gerçekleştiren bir POLY işlevini kullanarak her şeyi 16 bayta kadar hashler ve girişin boyutu büyüdükçe ana değer değişir.
  • Katman 3, 16 baytlık dizeyi 4 baytlık sabit bir uzunluğa hash eder. Bu, bir yinelemenin ürettiği şeydir.

İçinde RFC 4418 NH, aşağıdaki şekilde yeniden düzenlenmiştir:

Y = 0 için (i = 0; i 

Bu tanım, programcıları birikim üzerinde SIMD talimatlarını kullanmaya teşvik etmek için tasarlanmıştır, çünkü yalnızca dört endeksi uzakta olan veriler aynı SIMD yazmacına konulmayacaktır ve bu nedenle toplu olarak çoğalmaları daha hızlı olacaktır. Varsayımsal bir makinede, basitçe şu dile çevrilebilir:

Varsayımsal montaj
movq        $0,   regY  ; Y = 0movq        $0,   kayıt  ; i = 0döngü:Ekle         reg1, regM, kayıt ; reg1 = M + iEkle         reg2, regM, kayıtvldr.4x32   vec1, reg1       ; bellekten * reg1'den vec1'e 4x32bit değer yüklevldr.4x32   vec2, reg2vmul. 4x64   vec3, vec1, vec2 ; vec3 = vec1 * vec2uaddv.4x64  reg3, vec3       ; yatay olarak vec3'ü reg3'e toplaEkle         regY, regY, reg3 ; regY = regY + reg3Ekle         kayıt, kayıt, $8cmp         kayıt, regTjlt         döngü

Ayrıca bakınız

  • Poly1305 güçlü evrensel hashing tabanlı başka bir hızlı MAC'dir.

Referanslar

  1. ^ Thorup, Mikkel (2009). Doğrusal problama için dize hashlemesi. Proc. 20. ACM-SIAM Ayrık Algoritmalar Sempozyumu (SODA). s. 655–664. CiteSeerX  10.1.1.215.4253. doi:10.1137/1.9781611973068.72. Arşivlendi (PDF) 2013-10-12 tarihinde orjinalinden.Bölüm 5.3
  2. ^ a b Black, J .; Halevi, S .; Krawczyk, H .; Krovetz, T. (1999). UMAC: Hızlı ve Güvenli Mesaj Kimlik Doğrulaması (PDF). Kriptolojideki Gelişmeler (CRYPTO '99). Arşivlenen orijinal (PDF) 2012-03-10 tarihinde., Denklem 1 ve ayrıca bölüm 4.2 "NH'nin Tanımı".

Dış bağlantılar